mirror of
				https://github.com/nvbn/thefuck.git
				synced 2025-11-04 00:52:04 +00:00 
			
		
		
		
	Compare commits
	
		
			185 Commits
		
	
	
		
			merge-cd-r
			...
			3.8
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					c6af8409d9 | ||
| 
						 | 
					95e7d00aec | ||
| 
						 | 
					cdccf1881e | ||
| 
						 | 
					db6053b301 | ||
| 
						 | 
					183b70c8b8 | ||
| 
						 | 
					5e0cc8c703 | ||
| 
						 | 
					1aa2ec1795 | ||
| 
						 | 
					0c98053f74 | ||
| 
						 | 
					17b2fba48d | ||
| 
						 | 
					43886c38ff | ||
| 
						 | 
					9070748a86 | ||
| 
						 | 
					61de6f4a51 | ||
| 
						 | 
					d102af41d9 | ||
| 
						 | 
					b7002bb9f9 | ||
| 
						 | 
					18b4f5df6a | ||
| 
						 | 
					28153db4a8 | ||
| 
						 | 
					047a1a6072 | ||
| 
						 | 
					69db5c70e6 | ||
| 
						 | 
					fa1edd4bae | ||
| 
						 | 
					333c4b2a3f | ||
| 
						 | 
					b1f10642fa | ||
| 
						 | 
					047efd5575 | ||
| 
						 | 
					f604756cb7 | ||
| 
						 | 
					a27115bff1 | ||
| 
						 | 
					5d00b3bc25 | ||
| 
						 | 
					0cf4f5e8b0 | ||
| 
						 | 
					41707b80c6 | ||
| 
						 | 
					3a39deb485 | ||
| 
						 | 
					d4bc8cebf1 | ||
| 
						 | 
					6daf687237 | ||
| 
						 | 
					2207dd2668 | ||
| 
						 | 
					d1ab08a797 | ||
| 
						 | 
					51e89a36ef | ||
| 
						 | 
					39f7cc37eb | ||
| 
						 | 
					251b69b5a0 | ||
| 
						 | 
					8b1f078e27 | ||
| 
						 | 
					a47e84fa6b | ||
| 
						 | 
					bcab700215 | ||
| 
						 | 
					4fb99fd7a8 | ||
| 
						 | 
					bb5f6bb705 | ||
| 
						 | 
					d92765d5df | ||
| 
						 | 
					d8de5cfd20 | ||
| 
						 | 
					d73b14ce4b | ||
| 
						 | 
					04b83cf7e8 | ||
| 
						 | 
					019f0abf67 | ||
| 
						 | 
					c8d3dcd1bf | ||
| 
						 | 
					4f5b382df4 | ||
| 
						 | 
					77ad68b04b | ||
| 
						 | 
					6b0311181d | ||
| 
						 | 
					48808f93ac | ||
| 
						 | 
					7ce4307c87 | ||
| 
						 | 
					f7f0660114 | ||
| 
						 | 
					46f2351907 | ||
| 
						 | 
					e9de01fa41 | ||
| 
						 | 
					9b260eb239 | ||
| 
						 | 
					f20d4dbf85 | ||
| 
						 | 
					4af7dc2748 | ||
| 
						 | 
					c8e9606c7d | ||
| 
						 | 
					d71b9c2e62 | ||
| 
						 | 
					60497b9d04 | ||
| 
						 | 
					619af2638a | ||
| 
						 | 
					a1f15bfe5f | ||
| 
						 | 
					ce959b2a8b | ||
| 
						 | 
					3cc4940842 | ||
| 
						 | 
					470c0ef699 | ||
| 
						 | 
					4f95b3365a | ||
| 
						 | 
					bbfd53d718 | ||
| 
						 | 
					28d078708b | ||
| 
						 | 
					778d5f3e6e | ||
| 
						 | 
					c5acee54ea | ||
| 
						 | 
					c3e9c1bfc1 | ||
| 
						 | 
					6c25b33b9e | ||
| 
						 | 
					052f415d94 | ||
| 
						 | 
					3438d6dde7 | ||
| 
						 | 
					af0fe66a9f | ||
| 
						 | 
					fe07fcaa62 | ||
| 
						 | 
					b5dc7aab6d | ||
| 
						 | 
					a2ec5aa3ff | ||
| 
						 | 
					0b2bda9e85 | ||
| 
						 | 
					68c882cf1a | ||
| 
						 | 
					abe8fb84c6 | ||
| 
						 | 
					86b17eb570 | ||
| 
						 | 
					dcc0ce7ff3 | ||
| 
						 | 
					fee874cddc | ||
| 
						 | 
					2ce1c6bf90 | ||
| 
						 | 
					9b129fad08 | ||
| 
						 | 
					20adefbe7d | ||
| 
						 | 
					52ef1fa47d | ||
| 
						 | 
					94f8652175 | ||
| 
						 | 
					abe287a52b | ||
| 
						 | 
					60e19a054a | ||
| 
						 | 
					45add1e3d6 | ||
| 
						 | 
					934870f650 | ||
| 
						 | 
					738cc0c9a8 | ||
| 
						 | 
					cb6f964a30 | ||
| 
						 | 
					9c9a7343de | ||
| 
						 | 
					3803452980 | ||
| 
						 | 
					a19833d0c7 | ||
| 
						 | 
					084b907ac0 | ||
| 
						 | 
					bd5cf38271 | ||
| 
						 | 
					3c3d17e0ea | ||
| 
						 | 
					2f353498de | ||
| 
						 | 
					f0f49c1865 | ||
| 
						 | 
					20fff3142c | ||
| 
						 | 
					6e22b9ec6c | ||
| 
						 | 
					d53240b777 | ||
| 
						 | 
					cab933e7e6 | ||
| 
						 | 
					8b05f6d46f | ||
| 
						 | 
					ec64fbd5ea | ||
| 
						 | 
					4f9fb796c4 | ||
| 
						 | 
					be744f20ba | ||
| 
						 | 
					1b12cd85e9 | ||
| 
						 | 
					47df80f6b8 | ||
| 
						 | 
					a0ef0efe46 | ||
| 
						 | 
					25662ad737 | ||
| 
						 | 
					42b344676e | ||
| 
						 | 
					a3e1cb6718 | ||
| 
						 | 
					f249098336 | ||
| 
						 | 
					c3b1ba7637 | ||
| 
						 | 
					b65a9a0a4f | ||
| 
						 | 
					29c1d1efcf | ||
| 
						 | 
					0560f4ba8e | ||
| 
						 | 
					f9aa0e7c6b | ||
| 
						 | 
					b18a049886 | ||
| 
						 | 
					9192b555b5 | ||
| 
						 | 
					d750d3d6d1 | ||
| 
						 | 
					3ad953001d | ||
| 
						 | 
					3b4b87d8ed | ||
| 
						 | 
					6c3d67763a | ||
| 
						 | 
					959680d24d | ||
| 
						 | 
					b0adc7f2ca | ||
| 
						 | 
					fc05364233 | ||
| 
						 | 
					ad3db4ac67 | ||
| 
						 | 
					4a7b335d7c | ||
| 
						 | 
					465f6191b0 | ||
| 
						 | 
					b2836319ad | ||
| 
						 | 
					b3e9b36bd1 | ||
| 
						 | 
					ae2949cfa2 | ||
| 
						 | 
					1bb04b41eb | ||
| 
						 | 
					acd0b3e024 | ||
| 
						 | 
					7c5676491a | ||
| 
						 | 
					8feb722ed0 | ||
| 
						 | 
					c3ea2fd0c7 | ||
| 
						 | 
					b55464b2ea | ||
| 
						 | 
					8ddb61ae89 | ||
| 
						 | 
					fe91008a9c | ||
| 
						 | 
					7f777213c5 | ||
| 
						 | 
					89f868c115 | ||
| 
						 | 
					81f6a25abc | ||
| 
						 | 
					cc9af78787 | ||
| 
						 | 
					1fc3f1b5bf | ||
| 
						 | 
					45574d06c9 | ||
| 
						 | 
					dc23d67a42 | ||
| 
						 | 
					2fea0d4c60 | ||
| 
						 | 
					8c8abca8d5 | ||
| 
						 | 
					bd6ee68c03 | ||
| 
						 | 
					16533e85a7 | ||
| 
						 | 
					b3a19fe439 | ||
| 
						 | 
					959b96cf6e | ||
| 
						 | 
					f20311fa89 | ||
| 
						 | 
					a4c391096a | ||
| 
						 | 
					e71a3e0cdb | ||
| 
						 | 
					2d995d464f | ||
| 
						 | 
					280751b36e | ||
| 
						 | 
					0a6a3db65d | ||
| 
						 | 
					ecfc180280 | ||
| 
						 | 
					dae58211ba | ||
| 
						 | 
					5e9b2c56da | ||
| 
						 | 
					192ab0bfb0 | ||
| 
						 | 
					372e983459 | ||
| 
						 | 
					346cb99217 | ||
| 
						 | 
					bbfedb861f | ||
| 
						 | 
					f5b0e96747 | ||
| 
						 | 
					12a33f56bc | ||
| 
						 | 
					590fdba2aa | ||
| 
						 | 
					f374142bf8 | ||
| 
						 | 
					540ff7e16d | ||
| 
						 | 
					806dad18bf | ||
| 
						 | 
					8b416f269f | ||
| 
						 | 
					5e44fb22be | ||
| 
						 | 
					5389d0c106 | ||
| 
						 | 
					c970f190d2 | ||
| 
						 | 
					8f25c95f06 | ||
| 
						 | 
					4a48108c69 | ||
| 
						 | 
					f5e8fe954e | 
@@ -17,8 +17,9 @@ addons:
 | 
			
		||||
      - pandoc
 | 
			
		||||
      - git
 | 
			
		||||
install:
 | 
			
		||||
  - pip install coveralls
 | 
			
		||||
  - pip install -r requirements.txt
 | 
			
		||||
  - pip install -U pip
 | 
			
		||||
  - pip install -U coveralls
 | 
			
		||||
  - pip install -Ur requirements.txt
 | 
			
		||||
  - python setup.py develop
 | 
			
		||||
  - rm -rf build
 | 
			
		||||
script:
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										25
									
								
								CONTRIBUTING.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								CONTRIBUTING.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,25 @@
 | 
			
		||||
# Report issues
 | 
			
		||||
If you have any issue with The Fuck, sorry about that, but we will do what we
 | 
			
		||||
can to fix that. Actually, maybe we already have, so first thing to do is to
 | 
			
		||||
update The Fuck and see if the bug is still there.
 | 
			
		||||
 | 
			
		||||
If it is (sorry again), check if the problem has not already been reported and
 | 
			
		||||
if not, just open an issue on [GitHub](https://github.com/nvbn/thefuck) with
 | 
			
		||||
the following basic information:
 | 
			
		||||
  - the output of `thefuck --version` (something like `The Fuck 3.1 using
 | 
			
		||||
    Python 3.5.0`);
 | 
			
		||||
  - your shell and its version (`bash`, `zsh`, *Windows PowerShell*, etc.);
 | 
			
		||||
  - your system (Debian 7, ArchLinux, Windows, etc.);
 | 
			
		||||
  - how to reproduce the bug;
 | 
			
		||||
  - the output of The Fuck with `THEFUCK_DEBUG=true` exported (typically execute
 | 
			
		||||
    `export THEFUCK_DEBUG=true` in your shell before The Fuck);
 | 
			
		||||
  - if the bug only appears with a specific application, the output of that
 | 
			
		||||
    application and its version;
 | 
			
		||||
  - anything else you think is relevant.
 | 
			
		||||
 | 
			
		||||
It's only with enough information that we can do something to fix the problem.
 | 
			
		||||
 | 
			
		||||
# Make a pull request
 | 
			
		||||
We gladly accept pull request on the [official
 | 
			
		||||
repository](https://github.com/nvbn/thefuck) for new rules, new features, bug
 | 
			
		||||
fixes, etc.
 | 
			
		||||
							
								
								
									
										58
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										58
									
								
								README.md
									
									
									
									
									
								
							@@ -1,10 +1,10 @@
 | 
			
		||||
# The Fuck [](https://travis-ci.org/nvbn/thefuck) 
 | 
			
		||||
# The Fuck [![Version][version-badge]][version-link] [![Build Status][travis-badge]][travis-link] [![Windows Build Status][appveyor-badge]][appveyor-link] [![Coverage][coverage-badge]][coverage-link] [![MIT License][license-badge]](LICENSE.md)
 | 
			
		||||
 | 
			
		||||
Magnificent app which corrects your previous console command,
 | 
			
		||||
inspired by a [@liamosaur](https://twitter.com/liamosaur/)
 | 
			
		||||
[tweet](https://twitter.com/liamosaur/status/506975850596536320).
 | 
			
		||||
 | 
			
		||||
[](https://raw.githubusercontent.com/nvbn/thefuck/master/example.gif)
 | 
			
		||||
[![gif with examples][examples-link]][examples-link]
 | 
			
		||||
 | 
			
		||||
Few more examples:
 | 
			
		||||
 | 
			
		||||
@@ -52,7 +52,7 @@ Python 3.4.2 (default, Oct  8 2014, 13:08:17)
 | 
			
		||||
git: 'brnch' is not a git command. See 'git --help'.
 | 
			
		||||
 | 
			
		||||
Did you mean this?
 | 
			
		||||
	branch
 | 
			
		||||
    branch
 | 
			
		||||
 | 
			
		||||
➜ fuck
 | 
			
		||||
git branch [enter/↑/↓/ctrl+c]
 | 
			
		||||
@@ -97,7 +97,7 @@ Reading package lists... Done
 | 
			
		||||
## Installation [*experimental*]
 | 
			
		||||
 | 
			
		||||
On Ubuntu and OS X you can install `The Fuck` with installation script:
 | 
			
		||||
 
 | 
			
		||||
 | 
			
		||||
```bash
 | 
			
		||||
wget -O - https://raw.githubusercontent.com/nvbn/thefuck/master/install.sh | sh - && $0
 | 
			
		||||
```
 | 
			
		||||
@@ -107,7 +107,7 @@ wget -O - https://raw.githubusercontent.com/nvbn/thefuck/master/install.sh | sh
 | 
			
		||||
Install `The Fuck` with `pip`:
 | 
			
		||||
 | 
			
		||||
```bash
 | 
			
		||||
sudo pip install thefuck
 | 
			
		||||
sudo -H pip install thefuck
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
[Or using an OS package manager (OS X, Ubuntu, Arch).](https://github.com/nvbn/thefuck/wiki/Installation)
 | 
			
		||||
@@ -141,8 +141,8 @@ using the matched rule and runs it. Rules enabled by default are as follows:
 | 
			
		||||
 | 
			
		||||
* `cargo` – runs `cargo build` instead of `cargo`;
 | 
			
		||||
* `cargo_no_command` – fixes wrongs commands like `cargo buid`;
 | 
			
		||||
* `cd_correction` – spellchecks and correct failed cd commands, when it's not possible
 | 
			
		||||
creates directories before cd'ing into them;
 | 
			
		||||
* `cd_correction` – spellchecks and correct failed cd commands;
 | 
			
		||||
* `cd_mkdir` – creates directories before cd'ing into them;
 | 
			
		||||
* `cd_parent` – changes `cd..` to `cd ..`;
 | 
			
		||||
* `composer_not_command` – fixes composer command name;
 | 
			
		||||
* `cp_omitting_directory` – adds `-a` when you `cp` directory;
 | 
			
		||||
@@ -155,19 +155,25 @@ creates directories before cd'ing into them;
 | 
			
		||||
* `dry` – fixes repetitions like `git git push`;
 | 
			
		||||
* `fix_alt_space` – replaces Alt+Space with Space character;
 | 
			
		||||
* `fix_file` – opens a file with an error in your `$EDITOR`;
 | 
			
		||||
* `git_add` – fixes *"Did you forget to 'git add'?"*;
 | 
			
		||||
* `git_add` – fixes *"pathspec 'foo' did not match any file(s) known to git."*;
 | 
			
		||||
* `git_branch_delete` – changes `git branch -d` to `git branch -D`;
 | 
			
		||||
* `git_branch_exists` – offers `git branch -d foo`, `git branch -D foo` or `git checkout foo` when creating a branch that already exists;
 | 
			
		||||
* `git_branch_list` – catches `git branch list` in place of `git branch` and removes created branch;
 | 
			
		||||
* `git_checkout` – fixes branch name or creates new branch;
 | 
			
		||||
* `git_diff_staged` – adds `--staged` to previous `git diff` with unexpected output;
 | 
			
		||||
* `git_fix_stash` – fixes `git stash` commands (misspelled subcommand and missing `save`);
 | 
			
		||||
* `git_help_aliased` – fixes `git help <alias>` commands replacing <alias> with the aliased command;
 | 
			
		||||
* `git_not_command` – fixes wrong git commands like `git brnch`;
 | 
			
		||||
* `git_pull` – sets upstream before executing previous `git pull`;
 | 
			
		||||
* `git_pull_clone` – clones instead of pulling when the repo does not exist;
 | 
			
		||||
* `git_push` – adds `--set-upstream origin $branch` to previous failed `git push`;
 | 
			
		||||
* `git_push_pull` – runs `git pull` when `push` was rejected;
 | 
			
		||||
* `git_rm_recursive` – adds `-r` when you try to `rm` a directory;
 | 
			
		||||
* `git_remote_seturl_add` – runs `git remote add` when `git remote set_url` on nonexistant remote;
 | 
			
		||||
* `git_stash` – stashes you local modifications before rebasing or switching branch;
 | 
			
		||||
* `go_run` – appends `.go` extension when compiling/running Go programs
 | 
			
		||||
* `git_two_dashes` – adds a missing dash to commands like `git commit -amend` or `git rebase -continue`;
 | 
			
		||||
* `go_run` – appends `.go` extension when compiling/running Go programs;
 | 
			
		||||
* `grep_arguments_order` – fixes grep arguments order for situations like `grep -lir . test`;
 | 
			
		||||
* `grep_recursive` – adds `-r` when you trying to `grep` directory;
 | 
			
		||||
* `gulp_not_task` – fixes misspelled `gulp` tasks;
 | 
			
		||||
* `has_exists_script` – prepends `./` when script/binary exists;
 | 
			
		||||
@@ -176,6 +182,7 @@ creates directories before cd'ing into them;
 | 
			
		||||
* `java` – removes `.java` extension when running Java programs;
 | 
			
		||||
* `javac` – appends missing `.java` when compiling Java files;
 | 
			
		||||
* `lein_not_task` – fixes wrong `lein` tasks like `lein rpl`;
 | 
			
		||||
* `ln_no_hard_link` – catches hard link creation on directories, suggest symbolic link;
 | 
			
		||||
* `ls_lah` – adds `-lah` to `ls`;
 | 
			
		||||
* `man` – changes manual section;
 | 
			
		||||
* `man_no_space` – fixes man commands without spaces, for example `mandiff`;
 | 
			
		||||
@@ -183,6 +190,7 @@ creates directories before cd'ing into them;
 | 
			
		||||
* `mkdir_p` – adds `-p` when you trying to create directory without parent;
 | 
			
		||||
* `mvn_no_command` – adds `clean package` to `mvn`;
 | 
			
		||||
* `mvn_unknown_lifecycle_phase` – fixes misspelled lifecycle phases with `mvn`;
 | 
			
		||||
* `npm_wrong_command` – fixes wrong npm commands like `npm urgrade`;
 | 
			
		||||
* `no_command` – fixes wrong console commands, for example `vom/vim`;
 | 
			
		||||
* `no_such_file` – creates missing directories with `mv` and `cp` commands;
 | 
			
		||||
* `open` – prepends `http` to address passed to `open`;
 | 
			
		||||
@@ -210,6 +218,7 @@ Enabled by default only on specific platforms:
 | 
			
		||||
 | 
			
		||||
* `apt_get` – installs app from apt if it not installed (requires `python-commandnotfound` / `python3-commandnotfound`);
 | 
			
		||||
* `apt_get_search` – changes trying to search using `apt-get` with searching using `apt-cache`;
 | 
			
		||||
* `apt_invalid_operation` – fixes invalid `apt` and `apt-get` calls, like `apt-get isntall vim`;
 | 
			
		||||
* `brew_install` – fixes formula name for `brew install`;
 | 
			
		||||
* `brew_unknown_command` – fixes wrong brew commands, for example `brew docto/brew doctor`;
 | 
			
		||||
* `brew_upgrade` – appends `--all` to `brew upgrade` as per Homebrew's new behaviour;
 | 
			
		||||
@@ -218,13 +227,13 @@ Enabled by default only on specific platforms:
 | 
			
		||||
 | 
			
		||||
Bundled, but not enabled by default:
 | 
			
		||||
 | 
			
		||||
* `git_push_force` – adds `--force` to a `git push` (may conflict with `git_push_pull`);
 | 
			
		||||
* `git_push_force` – adds `--force-with-lease` to a `git push` (may conflict with `git_push_pull`);
 | 
			
		||||
* `rm_root` – adds `--no-preserve-root` to `rm -rf /` command.
 | 
			
		||||
 | 
			
		||||
## Creating your own rules
 | 
			
		||||
 | 
			
		||||
For adding your own rule you should create `your-rule-name.py`
 | 
			
		||||
in `~/.thefuck/rules`. The rule should contain two functions:
 | 
			
		||||
in `~/.config/thefuck/rules`. The rule should contain two functions:
 | 
			
		||||
 | 
			
		||||
```python
 | 
			
		||||
match(command: Command) -> bool
 | 
			
		||||
@@ -241,7 +250,7 @@ and optional `enabled_by_default`, `requires_output` and `priority` variables.
 | 
			
		||||
`Command` has three attributes: `script`, `stdout` and `stderr`.
 | 
			
		||||
 | 
			
		||||
*Rules api changed in 3.0:* For accessing settings in rule you need to import it with `from thefuck.conf import settings`.
 | 
			
		||||
`settings` is a special object filled with `~/.thefuck/settings.py` and values from env ([see more below](#settings)).
 | 
			
		||||
`settings` is a special object filled with `~/.config/thefuck/settings.py` and values from env ([see more below](#settings)).
 | 
			
		||||
 | 
			
		||||
Simple example of the rule for running script with `sudo`:
 | 
			
		||||
 | 
			
		||||
@@ -271,7 +280,7 @@ requires_output = True
 | 
			
		||||
 | 
			
		||||
## Settings
 | 
			
		||||
 | 
			
		||||
The Fuck has a few settings parameters which can be changed in `~/.thefuck/settings.py`:
 | 
			
		||||
The Fuck has a few settings parameters which can be changed in `$XDG_CONFIG_HOME/thefuck/settings.py` (`$XDG_CONFIG_HOME` defaults to `~/.config`):
 | 
			
		||||
 | 
			
		||||
* `rules` – list of enabled rules, by default `thefuck.conf.DEFAULT_RULES`;
 | 
			
		||||
* `exclude_rules` – list of disabled rules, by default `[]`;
 | 
			
		||||
@@ -279,7 +288,9 @@ The Fuck has a few settings parameters which can be changed in `~/.thefuck/setti
 | 
			
		||||
* `wait_command` – max amount of time in seconds for getting previous command output;
 | 
			
		||||
* `no_colors` – disable colored output;
 | 
			
		||||
* `priority` – dict with rules priorities, rule with lower `priority` will be matched first;
 | 
			
		||||
* `debug` – enables debug output, by default `False`.
 | 
			
		||||
* `debug` – enables debug output, by default `False`;
 | 
			
		||||
* `history_limit` – numeric value of how many history commands will be scanned, like `2000`;
 | 
			
		||||
* `alter_history` – push fixed command to history, by default `True`.
 | 
			
		||||
 | 
			
		||||
Example of `settings.py`:
 | 
			
		||||
 | 
			
		||||
@@ -296,13 +307,15 @@ debug = False
 | 
			
		||||
Or via environment variables:
 | 
			
		||||
 | 
			
		||||
* `THEFUCK_RULES` – list of enabled rules, like `DEFAULT_RULES:rm_root` or `sudo:no_command`;
 | 
			
		||||
* `THEFUCK_EXCLUDE_RULES` – list of disabled rules, like `git_pull:git_push`; 
 | 
			
		||||
* `THEFUCK_EXCLUDE_RULES` – list of disabled rules, like `git_pull:git_push`;
 | 
			
		||||
* `THEFUCK_REQUIRE_CONFIRMATION` – require confirmation before running new command, `true/false`;
 | 
			
		||||
* `THEFUCK_WAIT_COMMAND` – max amount of time in seconds for getting previous command output;
 | 
			
		||||
* `THEFUCK_NO_COLORS` – disable colored output, `true/false`;
 | 
			
		||||
* `THEFUCK_PRIORITY` – priority of the rules, like `no_command=9999:apt_get=100`,
 | 
			
		||||
rule with lower `priority` will be matched first;
 | 
			
		||||
* `THEFUCK_DEBUG` – enables debug output, `true/false`.
 | 
			
		||||
* `THEFUCK_DEBUG` – enables debug output, `true/false`;
 | 
			
		||||
* `THEFUCK_HISTORY_LIMIT` – how many history commands will be scanned, like `2000`;
 | 
			
		||||
* `THEFUCK_ALTER_HISTORY` – push fixed command to history `true/false`.
 | 
			
		||||
 | 
			
		||||
For example:
 | 
			
		||||
 | 
			
		||||
@@ -313,6 +326,7 @@ export THEFUCK_REQUIRE_CONFIRMATION='true'
 | 
			
		||||
export THEFUCK_WAIT_COMMAND=10
 | 
			
		||||
export THEFUCK_NO_COLORS='false'
 | 
			
		||||
export THEFUCK_PRIORITY='no_command=9999:apt_get=100'
 | 
			
		||||
export THEFUCK_HISTORY_LIMIT='2000'
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Developing
 | 
			
		||||
@@ -345,3 +359,15 @@ sudo apt-get install pandoc
 | 
			
		||||
 | 
			
		||||
## License MIT
 | 
			
		||||
Project License can be found [here](LICENSE.md).
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
[version-badge]:   https://img.shields.io/pypi/v/thefuck.svg?label=version
 | 
			
		||||
[version-link]:    https://pypi.python.org/pypi/thefuck/
 | 
			
		||||
[travis-badge]:    https://img.shields.io/travis/nvbn/thefuck.svg
 | 
			
		||||
[travis-link]:     https://travis-ci.org/nvbn/thefuck
 | 
			
		||||
[appveyor-badge]:  https://img.shields.io/appveyor/ci/nvbn/thefuck.svg?label=windows%20build
 | 
			
		||||
[appveyor-link]:   https://ci.appveyor.com/project/nvbn/thefuck
 | 
			
		||||
[coverage-badge]:  https://img.shields.io/coveralls/nvbn/thefuck.svg
 | 
			
		||||
[coverage-link]:   https://coveralls.io/github/nvbn/thefuck
 | 
			
		||||
[license-badge]:   https://img.shields.io/badge/license-MIT-007EC7.svg
 | 
			
		||||
[examples-link]:   https://raw.githubusercontent.com/nvbn/thefuck/master/example.gif
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										22
									
								
								appveyor.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								appveyor.yml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,22 @@
 | 
			
		||||
build: false
 | 
			
		||||
 | 
			
		||||
environment:
 | 
			
		||||
  matrix:
 | 
			
		||||
    - PYTHON: "C:/Python27"
 | 
			
		||||
    - PYTHON: "C:/Python33"
 | 
			
		||||
    - PYTHON: "C:/Python34"
 | 
			
		||||
    - PYTHON: "C:/Python35"
 | 
			
		||||
 | 
			
		||||
init:
 | 
			
		||||
  - "ECHO %PYTHON%"
 | 
			
		||||
  - ps: "ls C:/Python*"
 | 
			
		||||
 | 
			
		||||
install:
 | 
			
		||||
  - ps: (new-object net.webclient).DownloadFile('https://bootstrap.pypa.io/get-pip.py', 'C:/get-pip.py')
 | 
			
		||||
  - "%PYTHON%/python.exe C:/get-pip.py"
 | 
			
		||||
  - "%PYTHON%/Scripts/pip.exe install -U setuptools"
 | 
			
		||||
  - "%PYTHON%/python.exe setup.py develop"
 | 
			
		||||
  - "%PYTHON%/Scripts/pip.exe install -U -r requirements.txt"
 | 
			
		||||
 | 
			
		||||
test_script:
 | 
			
		||||
  - "%PYTHON%/Scripts/py.test.exe -sv"
 | 
			
		||||
							
								
								
									
										80
									
								
								install.sh
									
									
									
									
									
								
							
							
						
						
									
										80
									
								
								install.sh
									
									
									
									
									
								
							@@ -8,50 +8,54 @@ installed () {
 | 
			
		||||
    hash $1 2>/dev/null
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
# Install os dependencies:
 | 
			
		||||
if installed apt-get; then
 | 
			
		||||
    # Debian/ubuntu:
 | 
			
		||||
    sudo apt-get update -yy
 | 
			
		||||
    sudo apt-get install -yy python-pip python-dev command-not-found
 | 
			
		||||
install_thefuck () {
 | 
			
		||||
    # Install OS dependencies:
 | 
			
		||||
    if installed apt-get; then
 | 
			
		||||
        # Debian/Ubuntu:
 | 
			
		||||
        sudo apt-get update -yy
 | 
			
		||||
        sudo apt-get install -yy python-pip python-dev command-not-found python-gdbm
 | 
			
		||||
 | 
			
		||||
    if [[ -n $(apt-cache search python-commandnotfound) ]]; then
 | 
			
		||||
        # In case of different python versions:
 | 
			
		||||
        sudo apt-get install -yy python-commandnotfound
 | 
			
		||||
    fi
 | 
			
		||||
else
 | 
			
		||||
    if installed brew; then
 | 
			
		||||
        # OS X:
 | 
			
		||||
        brew update
 | 
			
		||||
        brew install python
 | 
			
		||||
        if [ -n "$(apt-cache search python-commandnotfound)" ]; then
 | 
			
		||||
            # In case of different python versions:
 | 
			
		||||
            sudo apt-get install -yy python-commandnotfound
 | 
			
		||||
        fi
 | 
			
		||||
    else
 | 
			
		||||
        # Genreic way:
 | 
			
		||||
        wget https://bootstrap.pypa.io/get-pip.py
 | 
			
		||||
        sudo python get-pip.py
 | 
			
		||||
        rm get-pip.py
 | 
			
		||||
        if installed brew; then
 | 
			
		||||
            # OS X:
 | 
			
		||||
            brew update
 | 
			
		||||
            brew install python
 | 
			
		||||
        else
 | 
			
		||||
            # Generic way:
 | 
			
		||||
            wget https://bootstrap.pypa.io/get-pip.py
 | 
			
		||||
            sudo python get-pip.py
 | 
			
		||||
            rm get-pip.py
 | 
			
		||||
        fi
 | 
			
		||||
    fi
 | 
			
		||||
fi
 | 
			
		||||
 | 
			
		||||
# thefuck requires fresh versions of setuptools and pip:
 | 
			
		||||
sudo pip install -U pip setuptools
 | 
			
		||||
sudo pip install -U thefuck
 | 
			
		||||
    # thefuck requires fresh versions of setuptools and pip:
 | 
			
		||||
    sudo pip install -U pip setuptools
 | 
			
		||||
    sudo pip install -U thefuck
 | 
			
		||||
 | 
			
		||||
# Setup aliases:
 | 
			
		||||
if should_add_alias ~/.bashrc; then
 | 
			
		||||
    echo 'eval $(thefuck --alias)' >> ~/.bashrc
 | 
			
		||||
fi
 | 
			
		||||
    # Setup aliases:
 | 
			
		||||
    if should_add_alias ~/.bashrc; then
 | 
			
		||||
        echo 'eval $(thefuck --alias)' >> ~/.bashrc
 | 
			
		||||
    fi
 | 
			
		||||
 | 
			
		||||
if should_add_alias ~/.bash_profile; then
 | 
			
		||||
    echo 'eval $(thefuck --alias)' >> ~/.bash_profile
 | 
			
		||||
fi
 | 
			
		||||
    if should_add_alias ~/.bash_profile; then
 | 
			
		||||
        echo 'eval $(thefuck --alias)' >> ~/.bash_profile
 | 
			
		||||
    fi
 | 
			
		||||
 | 
			
		||||
if should_add_alias ~/.zshrc; then
 | 
			
		||||
    echo 'eval $(thefuck --alias)' >> ~/.zshrc
 | 
			
		||||
fi
 | 
			
		||||
    if should_add_alias ~/.zshrc; then
 | 
			
		||||
        echo 'eval $(thefuck --alias)' >> ~/.zshrc
 | 
			
		||||
    fi
 | 
			
		||||
 | 
			
		||||
if should_add_alias ~/.config/fish/config.fish; then
 | 
			
		||||
    thefuck --alias >> ~/.config/fish/config.fish
 | 
			
		||||
fi
 | 
			
		||||
    if should_add_alias ~/.config/fish/config.fish; then
 | 
			
		||||
        thefuck --alias >> ~/.config/fish/config.fish
 | 
			
		||||
    fi
 | 
			
		||||
 | 
			
		||||
if should_add_alias ~/.tcshrc; then
 | 
			
		||||
    echo 'eval `thefuck --alias`' >> ~/.tcshrc
 | 
			
		||||
fi
 | 
			
		||||
    if should_add_alias ~/.tcshrc; then
 | 
			
		||||
        echo 'eval `thefuck --alias`' >> ~/.tcshrc
 | 
			
		||||
    fi
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
install_thefuck
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,4 @@
 | 
			
		||||
pip
 | 
			
		||||
pytest
 | 
			
		||||
mock
 | 
			
		||||
pytest-mock
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										13
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								setup.py
									
									
									
									
									
								
							@@ -1,8 +1,14 @@
 | 
			
		||||
#!/usr/bin/env python
 | 
			
		||||
from setuptools import setup, find_packages
 | 
			
		||||
import pip
 | 
			
		||||
import sys
 | 
			
		||||
import os
 | 
			
		||||
 | 
			
		||||
if int(pip.__version__.split('.')[0]) < 6:
 | 
			
		||||
    print('pip older than 6.0 not supported, please upgrade pip with:\n\n'
 | 
			
		||||
          '    pip install -U pip')
 | 
			
		||||
    sys.exit(-1)
 | 
			
		||||
 | 
			
		||||
if os.environ.get('CONVERT_README'):
 | 
			
		||||
    import pypandoc
 | 
			
		||||
 | 
			
		||||
@@ -20,10 +26,11 @@ elif (3, 0) < version < (3, 3):
 | 
			
		||||
          ' ({}.{} detected).'.format(*version))
 | 
			
		||||
    sys.exit(-1)
 | 
			
		||||
 | 
			
		||||
VERSION = '3.1'
 | 
			
		||||
VERSION = '3.8'
 | 
			
		||||
 | 
			
		||||
install_requires = ['psutil', 'colorama', 'six', 'decorator']
 | 
			
		||||
extras_require = {':python_version<"3.4"': ['pathlib']}
 | 
			
		||||
extras_require = {':python_version<"3.4"': ['pathlib'],
 | 
			
		||||
                  ":sys_platform=='win32'": ['win_unicode_console']}
 | 
			
		||||
 | 
			
		||||
setup(name='thefuck',
 | 
			
		||||
      version=VERSION,
 | 
			
		||||
@@ -34,7 +41,7 @@ setup(name='thefuck',
 | 
			
		||||
      url='https://github.com/nvbn/thefuck',
 | 
			
		||||
      license='MIT',
 | 
			
		||||
      packages=find_packages(exclude=['ez_setup', 'examples',
 | 
			
		||||
                                      'tests', 'release']),
 | 
			
		||||
                                      'tests', 'tests.*', 'release']),
 | 
			
		||||
      include_package_data=True,
 | 
			
		||||
      zip_safe=False,
 | 
			
		||||
      install_requires=install_requires,
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,9 @@
 | 
			
		||||
from pathlib import Path
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck import conf
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck import conf, const
 | 
			
		||||
 | 
			
		||||
shells.shell = shells.Generic()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def pytest_addoption(parser):
 | 
			
		||||
@@ -19,7 +22,7 @@ def no_memoize(monkeypatch):
 | 
			
		||||
def settings(request):
 | 
			
		||||
    def _reset_settings():
 | 
			
		||||
        conf.settings.clear()
 | 
			
		||||
        conf.settings.update(conf.DEFAULT_SETTINGS)
 | 
			
		||||
        conf.settings.update(const.DEFAULT_SETTINGS)
 | 
			
		||||
 | 
			
		||||
    request.addfinalizer(_reset_settings)
 | 
			
		||||
    conf.settings.user_dir = Path('~/.thefuck')
 | 
			
		||||
@@ -46,3 +49,14 @@ def functional(request):
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def source_root():
 | 
			
		||||
    return Path(__file__).parent.parent.resolve()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def set_shell(monkeypatch, request):
 | 
			
		||||
    def _set(cls):
 | 
			
		||||
        shell = cls()
 | 
			
		||||
        monkeypatch.setattr('thefuck.shells.shell', shell)
 | 
			
		||||
        request.addfinalizer()
 | 
			
		||||
        return shell
 | 
			
		||||
 | 
			
		||||
    return _set
 | 
			
		||||
 
 | 
			
		||||
@@ -49,7 +49,7 @@ def test_select_command_with_arrows(proc, TIMEOUT):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.functional
 | 
			
		||||
@pytest.mark.once_without_docker
 | 
			
		||||
@pytest.mark.skip_without_docker
 | 
			
		||||
def test_refuse_with_confirmation(proc, TIMEOUT):
 | 
			
		||||
    refuse_with_confirmation(proc, TIMEOUT)
 | 
			
		||||
    history_not_changed(proc, TIMEOUT)
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +0,0 @@
 | 
			
		||||
import pytest
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture(autouse=True)
 | 
			
		||||
def generic_shell(monkeypatch):
 | 
			
		||||
    monkeypatch.setattr('thefuck.shells.and_', lambda *x: ' && '.join(x))
 | 
			
		||||
@@ -29,12 +29,27 @@ def test_match_mocked(cmdnf_mock, command, return_value):
 | 
			
		||||
    assert get_packages.called
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# python-commandnotfound is available in ubuntu 14.04+
 | 
			
		||||
@pytest.mark.skipif(not getattr(apt_get, 'enabled_by_default', True),
 | 
			
		||||
                    reason='Skip if python-commandnotfound is not available')
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='a_bad_cmd', stderr='a_bad_cmd: command not found'),
 | 
			
		||||
    Command(script='vim', stderr=''), Command()])
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command, return_value', [
 | 
			
		||||
    (Command(script='a_bad_cmd', stderr='a_bad_cmd: command not found'), []),
 | 
			
		||||
    (Command(script='vim', stderr=''), []), (Command(), [])])
 | 
			
		||||
@patch('thefuck.rules.apt_get.CommandNotFound', create=True)
 | 
			
		||||
@patch.multiple(apt_get, create=True, apt_get='apt_get')
 | 
			
		||||
def test_not_match_mocked(cmdnf_mock, command, return_value):
 | 
			
		||||
    get_packages = Mock(return_value=return_value)
 | 
			
		||||
    cmdnf_mock.CommandNotFound.return_value = Mock(getPackages=get_packages)
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# python-commandnotfound is available in ubuntu 14.04+
 | 
			
		||||
@pytest.mark.skipif(not getattr(apt_get, 'enabled_by_default', True),
 | 
			
		||||
                    reason='Skip if python-commandnotfound is not available')
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										122
									
								
								tests/rules/test_apt_invalid_operation.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										122
									
								
								tests/rules/test_apt_invalid_operation.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,122 @@
 | 
			
		||||
from io import BytesIO
 | 
			
		||||
import pytest
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
from thefuck.rules.apt_invalid_operation import match, get_new_command, \
 | 
			
		||||
    _get_operations
 | 
			
		||||
 | 
			
		||||
invalid_operation = 'E: Invalid operation {}'.format
 | 
			
		||||
apt_help = b'''apt 1.0.10.2ubuntu1 for amd64 compiled on Oct  5 2015 15:55:05
 | 
			
		||||
Usage: apt [options] command
 | 
			
		||||
 | 
			
		||||
CLI for apt.
 | 
			
		||||
Basic commands:
 | 
			
		||||
 list - list packages based on package names
 | 
			
		||||
 search - search in package descriptions
 | 
			
		||||
 show - show package details
 | 
			
		||||
 | 
			
		||||
 update - update list of available packages
 | 
			
		||||
 | 
			
		||||
 install - install packages
 | 
			
		||||
 remove  - remove packages
 | 
			
		||||
 | 
			
		||||
 upgrade - upgrade the system by installing/upgrading packages
 | 
			
		||||
 full-upgrade - upgrade the system by removing/installing/upgrading packages
 | 
			
		||||
 | 
			
		||||
 edit-sources - edit the source information file
 | 
			
		||||
'''
 | 
			
		||||
apt_operations = ['list', 'search', 'show', 'update', 'install', 'remove',
 | 
			
		||||
                  'upgrade', 'full-upgrade', 'edit-sources']
 | 
			
		||||
 | 
			
		||||
apt_get_help = b'''apt 1.0.10.2ubuntu1 for amd64 compiled on Oct  5 2015 15:55:05
 | 
			
		||||
Usage: apt-get [options] command
 | 
			
		||||
       apt-get [options] install|remove pkg1 [pkg2 ...]
 | 
			
		||||
       apt-get [options] source pkg1 [pkg2 ...]
 | 
			
		||||
 | 
			
		||||
apt-get is a simple command line interface for downloading and
 | 
			
		||||
installing packages. The most frequently used commands are update
 | 
			
		||||
and install.
 | 
			
		||||
 | 
			
		||||
Commands:
 | 
			
		||||
   update - Retrieve new lists of packages
 | 
			
		||||
   upgrade - Perform an upgrade
 | 
			
		||||
   install - Install new packages (pkg is libc6 not libc6.deb)
 | 
			
		||||
   remove - Remove packages
 | 
			
		||||
   autoremove - Remove automatically all unused packages
 | 
			
		||||
   purge - Remove packages and config files
 | 
			
		||||
   source - Download source archives
 | 
			
		||||
   build-dep - Configure build-dependencies for source packages
 | 
			
		||||
   dist-upgrade - Distribution upgrade, see apt-get(8)
 | 
			
		||||
   dselect-upgrade - Follow dselect selections
 | 
			
		||||
   clean - Erase downloaded archive files
 | 
			
		||||
   autoclean - Erase old downloaded archive files
 | 
			
		||||
   check - Verify that there are no broken dependencies
 | 
			
		||||
   changelog - Download and display the changelog for the given package
 | 
			
		||||
   download - Download the binary package into the current directory
 | 
			
		||||
 | 
			
		||||
Options:
 | 
			
		||||
  -h  This help text.
 | 
			
		||||
  -q  Loggable output - no progress indicator
 | 
			
		||||
  -qq No output except for errors
 | 
			
		||||
  -d  Download only - do NOT install or unpack archives
 | 
			
		||||
  -s  No-act. Perform ordering simulation
 | 
			
		||||
  -y  Assume Yes to all queries and do not prompt
 | 
			
		||||
  -f  Attempt to correct a system with broken dependencies in place
 | 
			
		||||
  -m  Attempt to continue if archives are unlocatable
 | 
			
		||||
  -u  Show a list of upgraded packages as well
 | 
			
		||||
  -b  Build the source package after fetching it
 | 
			
		||||
  -V  Show verbose version numbers
 | 
			
		||||
  -c=? Read this configuration file
 | 
			
		||||
  -o=? Set an arbitrary configuration option, eg -o dir::cache=/tmp
 | 
			
		||||
See the apt-get(8), sources.list(5) and apt.conf(5) manual
 | 
			
		||||
pages for more information and options.
 | 
			
		||||
                       This APT has Super Cow Powers.
 | 
			
		||||
'''
 | 
			
		||||
apt_get_operations = ['update', 'upgrade', 'install', 'remove', 'autoremove',
 | 
			
		||||
                      'purge', 'source', 'build-dep', 'dist-upgrade',
 | 
			
		||||
                      'dselect-upgrade', 'clean', 'autoclean', 'check',
 | 
			
		||||
                      'changelog', 'download']
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stderr', [
 | 
			
		||||
    ('apt', invalid_operation('saerch')),
 | 
			
		||||
    ('apt-get', invalid_operation('isntall')),
 | 
			
		||||
    ('apt-cache', invalid_operation('rumove'))])
 | 
			
		||||
def test_match(script, stderr):
 | 
			
		||||
    assert match(Command(script, stderr=stderr))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stderr', [
 | 
			
		||||
    ('vim', invalid_operation('vim')),
 | 
			
		||||
    ('apt-get', "")])
 | 
			
		||||
def test_not_match(script, stderr):
 | 
			
		||||
    assert not match(Command(script, stderr=stderr))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def set_help(mocker):
 | 
			
		||||
    mock = mocker.patch('subprocess.Popen')
 | 
			
		||||
 | 
			
		||||
    def _set_text(text):
 | 
			
		||||
        mock.return_value.stdout = BytesIO(text)
 | 
			
		||||
 | 
			
		||||
    return _set_text
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('app, help_text, operations', [
 | 
			
		||||
    ('apt', apt_help, apt_operations),
 | 
			
		||||
    ('apt-get', apt_get_help, apt_get_operations)
 | 
			
		||||
])
 | 
			
		||||
def test_get_operations(set_help, app, help_text, operations):
 | 
			
		||||
    set_help(help_text)
 | 
			
		||||
    assert _get_operations(app) == operations
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stderr, help_text, result', [
 | 
			
		||||
    ('apt-get isntall vim', invalid_operation('isntall'),
 | 
			
		||||
     apt_get_help, 'apt-get install vim'),
 | 
			
		||||
    ('apt saerch vim', invalid_operation('saerch'),
 | 
			
		||||
     apt_help, 'apt search vim'),
 | 
			
		||||
])
 | 
			
		||||
def test_get_new_command(set_help, stderr, script, help_text, result):
 | 
			
		||||
    set_help(help_text)
 | 
			
		||||
    assert get_new_command(Command(script, stderr=stderr))[0] == result
 | 
			
		||||
@@ -31,8 +31,7 @@ def test_match(brew_no_available_formula, brew_already_installed,
 | 
			
		||||
                         stderr=brew_no_available_formula))
 | 
			
		||||
    assert not match(Command('brew install git',
 | 
			
		||||
                             stderr=brew_already_installed))
 | 
			
		||||
    assert not match(Command('brew install', stderr=brew_install_no_argument),
 | 
			
		||||
                     None)
 | 
			
		||||
    assert not match(Command('brew install', stderr=brew_install_no_argument))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.skipif(_is_not_okay_to_test(),
 | 
			
		||||
@@ -43,5 +42,5 @@ def test_get_new_command(brew_no_available_formula):
 | 
			
		||||
        == 'brew install elasticsearch'
 | 
			
		||||
 | 
			
		||||
    assert get_new_command(Command('brew install aa',
 | 
			
		||||
                                   stderr=brew_no_available_formula),
 | 
			
		||||
                           None) != 'brew install aha'
 | 
			
		||||
                                   stderr=brew_no_available_formula))\
 | 
			
		||||
        != 'brew install aha'
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.cd_correction import match, get_new_command
 | 
			
		||||
from thefuck.rules.cd_mkdir import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,7 +1,8 @@
 | 
			
		||||
import os
 | 
			
		||||
import pytest
 | 
			
		||||
import tarfile
 | 
			
		||||
from thefuck.rules.dirty_untar import match, get_new_command, side_effect
 | 
			
		||||
from thefuck.rules.dirty_untar import match, get_new_command, side_effect, \
 | 
			
		||||
                                      tar_extensions
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -32,34 +33,41 @@ def tar_error(tmpdir):
 | 
			
		||||
 | 
			
		||||
    return fixture
 | 
			
		||||
 | 
			
		||||
parametrize_filename = pytest.mark.parametrize('filename', [
 | 
			
		||||
    'foo.tar',
 | 
			
		||||
    'foo.tar.gz',
 | 
			
		||||
    'foo.tgz'])
 | 
			
		||||
parametrize_extensions = pytest.mark.parametrize('ext', tar_extensions)
 | 
			
		||||
 | 
			
		||||
# (filename as typed by the user, unquoted filename, quoted filename as per shells.quote)
 | 
			
		||||
parametrize_filename = pytest.mark.parametrize('filename, unquoted, quoted', [
 | 
			
		||||
    ('foo{}', 'foo{}', 'foo{}'),
 | 
			
		||||
    ('foo\ bar{}', 'foo bar{}', "'foo bar{}'"),
 | 
			
		||||
    ('"foo bar{}"', 'foo bar{}', "'foo bar{}'")])
 | 
			
		||||
 | 
			
		||||
parametrize_script = pytest.mark.parametrize('script, fixed', [
 | 
			
		||||
    ('tar xvf {}', 'mkdir -p foo && tar xvf {} -C foo'),
 | 
			
		||||
    ('tar -xvf {}', 'mkdir -p foo && tar -xvf {} -C foo'),
 | 
			
		||||
    ('tar --extract -f {}', 'mkdir -p foo && tar --extract -f {} -C foo')])
 | 
			
		||||
    ('tar xvf {}', 'mkdir -p {dir} && tar xvf {filename} -C {dir}'),
 | 
			
		||||
    ('tar -xvf {}', 'mkdir -p {dir} && tar -xvf {filename} -C {dir}'),
 | 
			
		||||
    ('tar --extract -f {}', 'mkdir -p {dir} && tar --extract -f {filename} -C {dir}')])
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@parametrize_extensions
 | 
			
		||||
@parametrize_filename
 | 
			
		||||
@parametrize_script
 | 
			
		||||
def test_match(tar_error, filename, script, fixed):
 | 
			
		||||
    tar_error(filename)
 | 
			
		||||
    assert match(Command(script=script.format(filename)))
 | 
			
		||||
def test_match(ext, tar_error, filename, unquoted, quoted, script, fixed):
 | 
			
		||||
    tar_error(unquoted.format(ext))
 | 
			
		||||
    assert match(Command(script=script.format(filename.format(ext))))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@parametrize_extensions
 | 
			
		||||
@parametrize_filename
 | 
			
		||||
@parametrize_script
 | 
			
		||||
def test_side_effect(tar_error, filename, script, fixed):
 | 
			
		||||
    tar_error(filename)
 | 
			
		||||
    side_effect(Command(script=script.format(filename)), None)
 | 
			
		||||
    assert set(os.listdir('.')) == {filename, 'd'}
 | 
			
		||||
def test_side_effect(ext, tar_error, filename, unquoted, quoted, script, fixed):
 | 
			
		||||
    tar_error(unquoted.format(ext))
 | 
			
		||||
    side_effect(Command(script=script.format(filename.format(ext))), None)
 | 
			
		||||
    assert set(os.listdir('.')) == {unquoted.format(ext), 'd'}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@parametrize_extensions
 | 
			
		||||
@parametrize_filename
 | 
			
		||||
@parametrize_script
 | 
			
		||||
def test_get_new_command(tar_error, filename, script, fixed):
 | 
			
		||||
    tar_error(filename)
 | 
			
		||||
    assert get_new_command(Command(script=script.format(filename))) == fixed.format(filename)
 | 
			
		||||
def test_get_new_command(ext, tar_error, filename, unquoted, quoted, script, fixed):
 | 
			
		||||
    tar_error(unquoted.format(ext))
 | 
			
		||||
    assert (get_new_command(Command(script=script.format(filename.format(ext))))
 | 
			
		||||
            == fixed.format(dir=quoted.format(''), filename=filename.format(ext)))
 | 
			
		||||
 
 | 
			
		||||
@@ -1,48 +1,72 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import os
 | 
			
		||||
import pytest
 | 
			
		||||
import zipfile
 | 
			
		||||
from thefuck.rules.dirty_unzip import match, get_new_command, side_effect
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
from unicodedata import normalize
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def zip_error(tmpdir):
 | 
			
		||||
    path = os.path.join(str(tmpdir), 'foo.zip')
 | 
			
		||||
    def zip_error_inner(filename):
 | 
			
		||||
        path = os.path.join(str(tmpdir), filename)
 | 
			
		||||
 | 
			
		||||
    def reset(path):
 | 
			
		||||
        with zipfile.ZipFile(path, 'w') as archive:
 | 
			
		||||
            archive.writestr('a', '1')
 | 
			
		||||
            archive.writestr('b', '2')
 | 
			
		||||
            archive.writestr('c', '3')
 | 
			
		||||
        def reset(path):
 | 
			
		||||
            with zipfile.ZipFile(path, 'w') as archive:
 | 
			
		||||
                archive.writestr('a', '1')
 | 
			
		||||
                archive.writestr('b', '2')
 | 
			
		||||
                archive.writestr('c', '3')
 | 
			
		||||
 | 
			
		||||
            archive.writestr('d/e', '4')
 | 
			
		||||
                archive.writestr('d/e', '4')
 | 
			
		||||
 | 
			
		||||
            archive.extractall()
 | 
			
		||||
                archive.extractall()
 | 
			
		||||
 | 
			
		||||
    os.chdir(str(tmpdir))
 | 
			
		||||
    reset(path)
 | 
			
		||||
        os.chdir(str(tmpdir))
 | 
			
		||||
        reset(path)
 | 
			
		||||
 | 
			
		||||
    assert set(os.listdir('.')) == {'foo.zip', 'a', 'b', 'c', 'd'}
 | 
			
		||||
    assert set(os.listdir('./d')) == {'e'}
 | 
			
		||||
        dir_list = os.listdir(u'.')
 | 
			
		||||
        if filename not in dir_list:
 | 
			
		||||
            filename = normalize('NFD', filename)
 | 
			
		||||
 | 
			
		||||
        assert set(dir_list) == {filename, 'a', 'b', 'c', 'd'}
 | 
			
		||||
        assert set(os.listdir('./d')) == {'e'}
 | 
			
		||||
    return zip_error_inner
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script', [
 | 
			
		||||
    'unzip foo',
 | 
			
		||||
    'unzip foo.zip'])
 | 
			
		||||
def test_match(zip_error, script):
 | 
			
		||||
@pytest.mark.parametrize('script,filename', [
 | 
			
		||||
    (u'unzip café', u'café.zip'),
 | 
			
		||||
    (u'unzip café.zip', u'café.zip'),
 | 
			
		||||
    (u'unzip foo', u'foo.zip'),
 | 
			
		||||
    (u'unzip foo.zip', u'foo.zip')])
 | 
			
		||||
def test_match(zip_error, script, filename):
 | 
			
		||||
    zip_error(filename)
 | 
			
		||||
    assert match(Command(script=script))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script', [
 | 
			
		||||
    'unzip foo',
 | 
			
		||||
    'unzip foo.zip'])
 | 
			
		||||
def test_side_effect(zip_error, script):
 | 
			
		||||
@pytest.mark.parametrize('script,filename', [
 | 
			
		||||
    (u'unzip café', u'café.zip'),
 | 
			
		||||
    (u'unzip café.zip', u'café.zip'),
 | 
			
		||||
    (u'unzip foo', u'foo.zip'),
 | 
			
		||||
    (u'unzip foo.zip', u'foo.zip')])
 | 
			
		||||
def test_side_effect(zip_error, script, filename):
 | 
			
		||||
    zip_error(filename)
 | 
			
		||||
    side_effect(Command(script=script), None)
 | 
			
		||||
    assert set(os.listdir('.')) == {'foo.zip', 'd'}
 | 
			
		||||
 | 
			
		||||
    dir_list = os.listdir(u'.')
 | 
			
		||||
    if filename not in set(dir_list):
 | 
			
		||||
        filename = normalize('NFD', filename)
 | 
			
		||||
 | 
			
		||||
    assert set(dir_list) == {filename, 'd'}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script,fixed', [
 | 
			
		||||
    ('unzip foo', 'unzip foo -d foo'),
 | 
			
		||||
    ('unzip foo.zip', 'unzip foo.zip -d foo')])
 | 
			
		||||
def test_get_new_command(zip_error, script, fixed):
 | 
			
		||||
@pytest.mark.parametrize('script,fixed,filename', [
 | 
			
		||||
    (u'unzip café', u"unzip café -d 'café'", u'café.zip'),
 | 
			
		||||
    (u'unzip foo', u'unzip foo -d foo', u'foo.zip'),
 | 
			
		||||
    (u"unzip foo\\ bar.zip", u"unzip foo\\ bar.zip -d 'foo bar'", u'foo.zip'),
 | 
			
		||||
    (u"unzip 'foo bar.zip'", u"unzip 'foo bar.zip' -d 'foo bar'", u'foo.zip'),
 | 
			
		||||
    (u'unzip foo.zip', u'unzip foo.zip -d foo', u'foo.zip')])
 | 
			
		||||
def test_get_new_command(zip_error, script, fixed, filename):
 | 
			
		||||
    zip_error(filename)
 | 
			
		||||
    assert get_new_command(Command(script=script)) == fixed
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,5 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import pytest
 | 
			
		||||
import os
 | 
			
		||||
from thefuck.rules.fix_file import match, get_new_command
 | 
			
		||||
@@ -87,6 +89,20 @@ Traceback (most recent call last):
 | 
			
		||||
TypeError: first argument must be string or compiled pattern
 | 
			
		||||
"""),
 | 
			
		||||
 | 
			
		||||
(u'python café.py', u'café.py', 8, None, '',
 | 
			
		||||
u"""
 | 
			
		||||
Traceback (most recent call last):
 | 
			
		||||
  File "café.py", line 8, in <module>
 | 
			
		||||
    match("foo")
 | 
			
		||||
  File "café.py", line 5, in match
 | 
			
		||||
    m = re.search(None, command)
 | 
			
		||||
  File "/usr/lib/python3.4/re.py", line 170, in search
 | 
			
		||||
    return _compile(pattern, flags).search(string)
 | 
			
		||||
  File "/usr/lib/python3.4/re.py", line 293, in _compile
 | 
			
		||||
    raise TypeError("first argument must be string or compiled pattern")
 | 
			
		||||
TypeError: first argument must be string or compiled pattern
 | 
			
		||||
"""),
 | 
			
		||||
 | 
			
		||||
('ruby a.rb', 'a.rb', 3, None, '',
 | 
			
		||||
"""
 | 
			
		||||
a.rb:3: syntax error, unexpected keyword_end
 | 
			
		||||
@@ -227,7 +243,7 @@ def test_get_new_command_with_settings(mocker, monkeypatch, test, settings):
 | 
			
		||||
 | 
			
		||||
    if test[3]:
 | 
			
		||||
        assert (get_new_command(cmd) ==
 | 
			
		||||
            'dummy_editor {} +{}:{} && {}'.format(test[1], test[2], test[3], test[0]))
 | 
			
		||||
            u'dummy_editor {} +{}:{} && {}'.format(test[1], test[2], test[3], test[0]))
 | 
			
		||||
    else:
 | 
			
		||||
        assert (get_new_command(cmd) ==
 | 
			
		||||
            'dummy_editor {} +{} && {}'.format(test[1], test[2], test[0]))
 | 
			
		||||
            u'dummy_editor {} +{} && {}'.format(test[1], test[2], test[0]))
 | 
			
		||||
 
 | 
			
		||||
@@ -4,36 +4,28 @@ from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def did_not_match(target, did_you_forget=True):
 | 
			
		||||
    error = ("error: pathspec '{}' did not match any "
 | 
			
		||||
             "file(s) known to git.".format(target))
 | 
			
		||||
    if did_you_forget:
 | 
			
		||||
        error = ("{}\nDid you forget to 'git add'?'".format(error))
 | 
			
		||||
    return error
 | 
			
		||||
def stderr(target):
 | 
			
		||||
    return ("error: pathspec '{}' did not match any "
 | 
			
		||||
            'file(s) known to git.'.format(target))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='git submodule update unknown',
 | 
			
		||||
            stderr=did_not_match('unknown')),
 | 
			
		||||
    Command(script='git commit unknown',
 | 
			
		||||
            stderr=did_not_match('unknown'))])  # Older versions of Git
 | 
			
		||||
def test_match(command):
 | 
			
		||||
    assert match(command)
 | 
			
		||||
@pytest.mark.parametrize('script, target', [
 | 
			
		||||
    ('git submodule update unknown', 'unknown'),
 | 
			
		||||
    ('git commit unknown', 'unknown')])
 | 
			
		||||
def test_match(stderr, script, target):
 | 
			
		||||
    assert match(Command(script=script, stderr=stderr))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='git submodule update known', stderr=('')),
 | 
			
		||||
    Command(script='git commit known', stderr=('')),
 | 
			
		||||
    Command(script='git commit unknown',  # Newer versions of Git
 | 
			
		||||
            stderr=did_not_match('unknown', False))])
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
@pytest.mark.parametrize('script', [
 | 
			
		||||
    'git submodule update known', 'git commit known'])
 | 
			
		||||
def test_not_match(script):
 | 
			
		||||
    assert not match(Command(script=script, stderr=''))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command, new_command', [
 | 
			
		||||
    (Command('git submodule update unknown', stderr=did_not_match('unknown')),
 | 
			
		||||
@pytest.mark.parametrize('script, target, new_command', [
 | 
			
		||||
    ('git submodule update unknown', 'unknown',
 | 
			
		||||
     'git add -- unknown && git submodule update unknown'),
 | 
			
		||||
    (Command('git commit unknown', stderr=did_not_match('unknown')),  # Old Git
 | 
			
		||||
    ('git commit unknown', 'unknown',
 | 
			
		||||
     'git add -- unknown && git commit unknown')])
 | 
			
		||||
def test_get_new_command(command, new_command):
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
def test_get_new_command(stderr, script, target, new_command):
 | 
			
		||||
    assert get_new_command(Command(script=script, stderr=stderr)) == new_command
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										33
									
								
								tests/rules/test_git_branch_exists.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								tests/rules/test_git_branch_exists.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,33 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.git_branch_exists import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def stderr(branch_name):
 | 
			
		||||
    return "fatal: A branch named '{}' already exists.".format(branch_name)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def new_command(branch_name):
 | 
			
		||||
    return [cmd.format(branch_name) for cmd in [
 | 
			
		||||
        'git branch -d {0} && git branch {0}',
 | 
			
		||||
        'git branch -D {0} && git branch {0}', 'git checkout {0}']]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, branch_name', [
 | 
			
		||||
    ('git branch foo', 'foo'),
 | 
			
		||||
    ('git branch bar', 'bar')])
 | 
			
		||||
def test_match(stderr, script, branch_name):
 | 
			
		||||
    assert match(Command(script=script, stderr=stderr))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script', ['git branch foo', 'git branch bar'])
 | 
			
		||||
def test_not_match(script):
 | 
			
		||||
    assert not match(Command(script=script, stderr=''))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, branch_name, ', [
 | 
			
		||||
    ('git branch foo', 'foo'), ('git branch bar', 'bar')])
 | 
			
		||||
def test_get_new_command(stderr, new_command, script, branch_name):
 | 
			
		||||
    assert get_new_command(Command(script=script, stderr=stderr)) == new_command
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck.rules.git_branch_list import match, get_new_command
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -16,4 +16,4 @@ def test_not_match():
 | 
			
		||||
 | 
			
		||||
def test_get_new_command():
 | 
			
		||||
    assert (get_new_command(Command('git branch list')) ==
 | 
			
		||||
            shells.and_('git branch --delete list', 'git branch'))
 | 
			
		||||
            shell.and_('git branch --delete list', 'git branch'))
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,6 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.git_checkout import match, get_new_command
 | 
			
		||||
from io import BytesIO
 | 
			
		||||
from thefuck.rules.git_checkout import match, get_branches, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -13,8 +14,10 @@ def did_not_match(target, did_you_forget=False):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def get_branches(mocker):
 | 
			
		||||
    return mocker.patch('thefuck.rules.git_checkout.get_branches')
 | 
			
		||||
def git_branch(mocker, branches):
 | 
			
		||||
    mock = mocker.patch('subprocess.Popen')
 | 
			
		||||
    mock.return_value.stdout = BytesIO(branches)
 | 
			
		||||
    return mock
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
@@ -33,21 +36,34 @@ def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('branches, branch_list', [
 | 
			
		||||
    (b'', []),
 | 
			
		||||
    (b'* master', ['master']),
 | 
			
		||||
    (b'  remotes/origin/master', ['master']),
 | 
			
		||||
    (b'  just-another-branch', ['just-another-branch']),
 | 
			
		||||
    (b'* master\n  just-another-branch', ['master', 'just-another-branch']),
 | 
			
		||||
    (b'* master\n  remotes/origin/master\n  just-another-branch',
 | 
			
		||||
     ['master', 'master', 'just-another-branch'])])
 | 
			
		||||
def test_get_branches(branches, branch_list, git_branch):
 | 
			
		||||
    git_branch(branches)
 | 
			
		||||
    assert list(get_branches()) == branch_list
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('branches, command, new_command', [
 | 
			
		||||
    ([],
 | 
			
		||||
    (b'',
 | 
			
		||||
     Command(script='git checkout unknown', stderr=did_not_match('unknown')),
 | 
			
		||||
     'git branch unknown && git checkout unknown'),
 | 
			
		||||
    ([],
 | 
			
		||||
    (b'',
 | 
			
		||||
     Command('git commit unknown', stderr=did_not_match('unknown')),
 | 
			
		||||
     'git branch unknown && git commit unknown'),
 | 
			
		||||
    (['test-random-branch-123'],
 | 
			
		||||
    (b'  test-random-branch-123',
 | 
			
		||||
     Command(script='git checkout tst-rdm-brnch-123',
 | 
			
		||||
             stderr=did_not_match('tst-rdm-brnch-123')),
 | 
			
		||||
     'git checkout test-random-branch-123'),
 | 
			
		||||
    (['test-random-branch-123'],
 | 
			
		||||
    (b'  test-random-branch-123',
 | 
			
		||||
     Command(script='git commit tst-rdm-brnch-123',
 | 
			
		||||
             stderr=did_not_match('tst-rdm-brnch-123')),
 | 
			
		||||
     'git commit test-random-branch-123')])
 | 
			
		||||
def test_get_new_command(branches, command, new_command, get_branches):
 | 
			
		||||
    get_branches.return_value = branches
 | 
			
		||||
def test_get_new_command(branches, command, new_command, git_branch):
 | 
			
		||||
    git_branch(branches)
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										24
									
								
								tests/rules/test_git_help_aliased.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								tests/rules/test_git_help_aliased.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,24 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.git_help_aliased import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stdout', [
 | 
			
		||||
    ('git help st', "`git st' is aliased to `status'"),
 | 
			
		||||
    ('git help ds', "`git ds' is aliased to `diff --staged'")])
 | 
			
		||||
def test_match(script, stdout):
 | 
			
		||||
    assert match(Command(script=script, stdout=stdout))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stdout', [
 | 
			
		||||
    ('git help status', "GIT-STATUS(1)...Git Manual...GIT-STATUS(1)"),
 | 
			
		||||
    ('git help diff', "GIT-DIFF(1)...Git Manual...GIT-DIFF(1)")])
 | 
			
		||||
def test_not_match(script, stdout):
 | 
			
		||||
    assert not match(Command(script=script, stdout=stdout))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stdout, new_command', [
 | 
			
		||||
    ('git help st', "`git st' is aliased to `status'", 'git help status'),
 | 
			
		||||
    ('git help ds', "`git ds' is aliased to `diff --staged'", 'git help diff')])
 | 
			
		||||
def test_get_new_command(script, stdout, new_command):
 | 
			
		||||
    assert get_new_command(Command(script=script, stdout=stdout)) == new_command
 | 
			
		||||
@@ -45,8 +45,8 @@ def test_not_match(command):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command, output', [
 | 
			
		||||
    (Command(script='git push', stderr=git_err), 'git push --force'),
 | 
			
		||||
    (Command(script='git push nvbn', stderr=git_err), 'git push --force nvbn'),
 | 
			
		||||
    (Command(script='git push nvbn master', stderr=git_err), 'git push --force nvbn master')])
 | 
			
		||||
    (Command(script='git push', stderr=git_err), 'git push --force-with-lease'),
 | 
			
		||||
    (Command(script='git push nvbn', stderr=git_err), 'git push --force-with-lease nvbn'),
 | 
			
		||||
    (Command(script='git push nvbn master', stderr=git_err), 'git push --force-with-lease nvbn master')])
 | 
			
		||||
def test_get_new_command(command, output):
 | 
			
		||||
    assert get_new_command(command) == output
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										26
									
								
								tests/rules/test_git_remote_seturl_add.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								tests/rules/test_git_remote_seturl_add.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,26 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.git_remote_seturl_add import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='git remote set-url origin url', stderr="fatal: No such remote")])
 | 
			
		||||
def test_match(command):
 | 
			
		||||
    assert match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command('git remote set-url origin url', stderr=""),
 | 
			
		||||
    Command('git remote add origin url'),
 | 
			
		||||
    Command('git remote remove origin'),
 | 
			
		||||
    Command('git remote prune origin'),
 | 
			
		||||
    Command('git remote set-branches origin branch')
 | 
			
		||||
    ])
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command, new_command', [
 | 
			
		||||
    (Command('git remote set-url origin git@github.com:nvbn/thefuck.git'),
 | 
			
		||||
     'git remote add origin git@github.com:nvbn/thefuck.git')])
 | 
			
		||||
def test_get_new_command(command, new_command):
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
							
								
								
									
										27
									
								
								tests/rules/test_git_rm_recursive.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								tests/rules/test_git_rm_recursive.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.git_rm_recursive import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def stderr(target):
 | 
			
		||||
    return "fatal: not removing '{}' recursively without -r".format(target)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, target', [
 | 
			
		||||
    ('git rm foo', 'foo'),
 | 
			
		||||
    ('git rm foo bar', 'foo bar')])
 | 
			
		||||
def test_match(stderr, script, target):
 | 
			
		||||
    assert match(Command(script=script, stderr=stderr))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script', ['git rm foo', 'git rm foo bar'])
 | 
			
		||||
def test_not_match(script):
 | 
			
		||||
    assert not match(Command(script=script, stderr=''))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, target, new_command', [
 | 
			
		||||
    ('git rm foo', 'foo', 'git rm -r foo'),
 | 
			
		||||
    ('git rm foo bar', 'foo bar', 'git rm -r foo bar')])
 | 
			
		||||
def test_get_new_command(stderr, script, target, new_command):
 | 
			
		||||
    assert get_new_command(Command(script=script, stderr=stderr)) == new_command
 | 
			
		||||
							
								
								
									
										47
									
								
								tests/rules/test_git_two_dashes.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								tests/rules/test_git_two_dashes.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,47 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.git_two_dashes import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def stderr(meant):
 | 
			
		||||
    return 'error: did you mean `%s` (with two dashes ?)' % meant
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='git add -patch', stderr=stderr('--patch')),
 | 
			
		||||
    Command(script='git checkout -patch', stderr=stderr('--patch')),
 | 
			
		||||
    Command(script='git commit -amend', stderr=stderr('--amend')),
 | 
			
		||||
    Command(script='git push -tags', stderr=stderr('--tags')),
 | 
			
		||||
    Command(script='git rebase -continue', stderr=stderr('--continue'))])
 | 
			
		||||
def test_match(command):
 | 
			
		||||
    assert match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='git add --patch'),
 | 
			
		||||
    Command(script='git checkout --patch'),
 | 
			
		||||
    Command(script='git commit --amend'),
 | 
			
		||||
    Command(script='git push --tags'),
 | 
			
		||||
    Command(script='git rebase --continue')])
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command, output', [
 | 
			
		||||
    (Command(script='git add -patch', stderr=stderr('--patch')),
 | 
			
		||||
        'git add --patch'),
 | 
			
		||||
    (Command(script='git checkout -patch', stderr=stderr('--patch')),
 | 
			
		||||
        'git checkout --patch'),
 | 
			
		||||
    (Command(script='git checkout -patch', stderr=stderr('--patch')),
 | 
			
		||||
        'git checkout --patch'),
 | 
			
		||||
    (Command(script='git init -bare', stderr=stderr('--bare')),
 | 
			
		||||
        'git init --bare'),
 | 
			
		||||
    (Command(script='git commit -amend', stderr=stderr('--amend')),
 | 
			
		||||
        'git commit --amend'),
 | 
			
		||||
    (Command(script='git push -tags', stderr=stderr('--tags')),
 | 
			
		||||
        'git push --tags'),
 | 
			
		||||
    (Command(script='git rebase -continue', stderr=stderr('--continue')),
 | 
			
		||||
        'git rebase --continue')])
 | 
			
		||||
def test_get_new_command(command, output):
 | 
			
		||||
    assert get_new_command(command) == output
 | 
			
		||||
							
								
								
									
										40
									
								
								tests/rules/test_grep_arguments_order.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								tests/rules/test_grep_arguments_order.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,40 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.grep_arguments_order import get_new_command, match
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
stderr = 'grep: {}: No such file or directory'.format
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture(autouse=True)
 | 
			
		||||
def os_path(monkeypatch):
 | 
			
		||||
    monkeypatch.setattr('os.path.isfile', lambda x: not x.startswith('-'))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, file', [
 | 
			
		||||
    ('grep test.py test', 'test'),
 | 
			
		||||
    ('grep -lir . test', 'test'),
 | 
			
		||||
    ('egrep test.py test', 'test'),
 | 
			
		||||
    ('egrep -lir . test', 'test')])
 | 
			
		||||
def test_match(script, file):
 | 
			
		||||
    assert match(Command(script, stderr=stderr(file)))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stderr', [
 | 
			
		||||
    ('cat test.py', stderr('test')),
 | 
			
		||||
    ('grep test test.py', ''),
 | 
			
		||||
    ('grep -lir test .', ''),
 | 
			
		||||
    ('egrep test test.py', ''),
 | 
			
		||||
    ('egrep -lir test .', '')])
 | 
			
		||||
def test_not_match(script, stderr):
 | 
			
		||||
    assert not match(Command(script, stderr=stderr))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stderr, result', [
 | 
			
		||||
    ('grep test.py test', stderr('test'), 'grep test test.py'),
 | 
			
		||||
    ('grep -lir . test', stderr('test'), 'grep -lir test .'),
 | 
			
		||||
    ('grep . test -lir', stderr('test'), 'grep test -lir .'),
 | 
			
		||||
    ('egrep test.py test', stderr('test'), 'egrep test test.py'),
 | 
			
		||||
    ('egrep -lir . test', stderr('test'), 'egrep -lir test .'),
 | 
			
		||||
    ('egrep . test -lir', stderr('test'), 'egrep test -lir .')])
 | 
			
		||||
def test_get_new_command(script, stderr, result):
 | 
			
		||||
    assert get_new_command(Command(script, stderr=stderr)) == result
 | 
			
		||||
@@ -1,12 +1,15 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
from thefuck.rules.grep_recursive import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def test_match():
 | 
			
		||||
    assert match(Command('grep blah .', stderr='grep: .: Is a directory'))
 | 
			
		||||
    assert match(Command(u'grep café .', stderr='grep: .: Is a directory'))
 | 
			
		||||
    assert not match(Command())
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def test_get_new_command():
 | 
			
		||||
    assert get_new_command(
 | 
			
		||||
        Command('grep blah .')) == 'grep -r blah .'
 | 
			
		||||
    assert get_new_command(Command('grep blah .')) == 'grep -r blah .'
 | 
			
		||||
    assert get_new_command(Command(u'grep café .')) == u'grep -r café .'
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,5 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from io import BytesIO
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
from thefuck.rules.gulp_not_task import match, get_new_command
 | 
			
		||||
 | 
			
		||||
@@ -22,7 +23,7 @@ def test_not_march(script, stdout):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def test_get_new_command(mocker):
 | 
			
		||||
    mocker.patch('thefuck.rules.gulp_not_task.get_gulp_tasks', return_value=[
 | 
			
		||||
        'serve', 'build', 'default'])
 | 
			
		||||
    mock = mocker.patch('subprocess.Popen')
 | 
			
		||||
    mock.return_value.stdout = BytesIO(b'serve \nbuild \ndefault \n')
 | 
			
		||||
    command = Command('gulp srve', stdout('srve'))
 | 
			
		||||
    assert get_new_command(command) == ['gulp serve', 'gulp default']
 | 
			
		||||
 
 | 
			
		||||
@@ -1,17 +1,18 @@
 | 
			
		||||
from mock import Mock, patch
 | 
			
		||||
from mock import patch
 | 
			
		||||
from thefuck.rules.has_exists_script import match, get_new_command
 | 
			
		||||
from ..utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def test_match():
 | 
			
		||||
    with patch('os.path.exists', return_value=True):
 | 
			
		||||
        assert match(Mock(script='main', stderr='main: command not found'))
 | 
			
		||||
        assert match(Mock(script='main --help',
 | 
			
		||||
        assert match(Command(script='main', stderr='main: command not found'))
 | 
			
		||||
        assert match(Command(script='main --help',
 | 
			
		||||
                          stderr='main: command not found'))
 | 
			
		||||
        assert not match(Mock(script='main', stderr=''))
 | 
			
		||||
        assert not match(Command(script='main', stderr=''))
 | 
			
		||||
 | 
			
		||||
    with patch('os.path.exists', return_value=False):
 | 
			
		||||
        assert not match(Mock(script='main', stderr='main: command not found'))
 | 
			
		||||
        assert not match(Command(script='main', stderr='main: command not found'))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def test_get_new_command():
 | 
			
		||||
    assert get_new_command(Mock(script='main --help')) == './main --help'
 | 
			
		||||
    assert get_new_command(Command(script='main --help')) == './main --help'
 | 
			
		||||
 
 | 
			
		||||
@@ -3,38 +3,23 @@ from thefuck.rules.history import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def history(mocker):
 | 
			
		||||
    return mocker.patch('thefuck.rules.history.get_history',
 | 
			
		||||
                        return_value=['le cat', 'fuck', 'ls cat',
 | 
			
		||||
                                      'diff x', 'nocommand x'])
 | 
			
		||||
@pytest.fixture(autouse=True)
 | 
			
		||||
def history_without_current(mocker):
 | 
			
		||||
    return mocker.patch(
 | 
			
		||||
        'thefuck.rules.history.get_valid_history_without_current',
 | 
			
		||||
        return_value=['ls cat', 'diff x'])
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def alias(mocker):
 | 
			
		||||
    return mocker.patch('thefuck.rules.history.thefuck_alias',
 | 
			
		||||
                        return_value='fuck')
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def callables(mocker):
 | 
			
		||||
    return mocker.patch('thefuck.rules.history.get_all_executables',
 | 
			
		||||
                        return_value=['diff', 'ls'])
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('history', 'callables', 'no_memoize', 'alias')
 | 
			
		||||
@pytest.mark.parametrize('script', ['ls cet', 'daff x'])
 | 
			
		||||
def test_match(script):
 | 
			
		||||
    assert match(Command(script=script))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('history', 'callables', 'no_memoize', 'alias')
 | 
			
		||||
@pytest.mark.parametrize('script', ['apt-get', 'nocommand y'])
 | 
			
		||||
def test_not_match(script):
 | 
			
		||||
    assert not match(Command(script=script))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('history', 'callables', 'no_memoize', 'alias')
 | 
			
		||||
@pytest.mark.parametrize('script, result', [
 | 
			
		||||
    ('ls cet', 'ls cat'),
 | 
			
		||||
    ('daff x', 'diff x')])
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										37
									
								
								tests/rules/test_ln_no_hard_link.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								tests/rules/test_ln_no_hard_link.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,37 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.ln_no_hard_link import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
error = "hard link not allowed for directory"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stderr', [
 | 
			
		||||
    ("ln barDir barLink", "ln: ‘barDir’: {}"),
 | 
			
		||||
    ("sudo ln a b", "ln: ‘a’: {}"),
 | 
			
		||||
    ("sudo ln -nbi a b", "ln: ‘a’: {}")])
 | 
			
		||||
def test_match(script, stderr):
 | 
			
		||||
    command = Command(script, stderr=stderr.format(error))
 | 
			
		||||
    assert match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stderr', [
 | 
			
		||||
    ('', ''),
 | 
			
		||||
    ("ln a b", "... hard link"),
 | 
			
		||||
    ("sudo ln a b", "... hard link"),
 | 
			
		||||
    ("a b", error)])
 | 
			
		||||
def test_assert_not_match(script, stderr):
 | 
			
		||||
    command = Command(script, stderr=stderr)
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, result', [
 | 
			
		||||
    ("ln barDir barLink", "ln -s barDir barLink"),
 | 
			
		||||
    ("sudo ln barDir barLink", "sudo ln -s barDir barLink"),
 | 
			
		||||
    ("sudo ln -nbi a b", "sudo ln -s -nbi a b"),
 | 
			
		||||
    ("ln -nbi a b && ls", "ln -s -nbi a b && ls"),
 | 
			
		||||
    ("ln a ln", "ln -s a ln"),
 | 
			
		||||
    ("sudo ln a ln", "sudo ln -s a ln")])
 | 
			
		||||
def test_get_new_command(script, result):
 | 
			
		||||
    command = Command(script)
 | 
			
		||||
    assert get_new_command(command) == result
 | 
			
		||||
@@ -7,7 +7,7 @@ from tests.utils import Command
 | 
			
		||||
    Command('mkdir foo/bar/baz', stderr='mkdir: foo/bar: No such file or directory'),
 | 
			
		||||
    Command('./bin/hdfs dfs -mkdir foo/bar/baz', stderr='mkdir: `foo/bar/baz\': No such file or directory'),
 | 
			
		||||
    Command('hdfs dfs -mkdir foo/bar/baz', stderr='mkdir: `foo/bar/baz\': No such file or directory')
 | 
			
		||||
    ])
 | 
			
		||||
])
 | 
			
		||||
def test_match(command):
 | 
			
		||||
    assert match(command)
 | 
			
		||||
 | 
			
		||||
@@ -17,7 +17,8 @@ def test_match(command):
 | 
			
		||||
    Command('mkdir foo/bar/baz', stderr='foo bar baz'),
 | 
			
		||||
    Command('hdfs dfs -mkdir foo/bar/baz'),
 | 
			
		||||
    Command('./bin/hdfs dfs -mkdir foo/bar/baz'),
 | 
			
		||||
    Command()])
 | 
			
		||||
    Command(),
 | 
			
		||||
])
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
@@ -25,7 +26,7 @@ def test_not_match(command):
 | 
			
		||||
@pytest.mark.parametrize('command, new_command', [
 | 
			
		||||
    (Command('mkdir foo/bar/baz'), 'mkdir -p foo/bar/baz'),
 | 
			
		||||
    (Command('hdfs dfs -mkdir foo/bar/baz'), 'hdfs dfs -mkdir -p foo/bar/baz'),
 | 
			
		||||
    (Command('./bin/hdfs dfs -mkdir foo/bar/baz'), './bin/hdfs dfs -mkdir -p foo/bar/baz')])
 | 
			
		||||
    (Command('./bin/hdfs dfs -mkdir foo/bar/baz'), './bin/hdfs dfs -mkdir -p foo/bar/baz'),
 | 
			
		||||
])
 | 
			
		||||
def test_get_new_command(command, new_command):
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -32,9 +32,9 @@ def test_match(command):
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command, new_command', [
 | 
			
		||||
    (Command(script='mvn', stdout='[ERROR] No goals have been specified for this build. You must specify a valid lifecycle phase or a goal in the format <plugin-prefix>:<goal> or <plugin-group-id>:<plugin-artifact-id>[:<plugin-version>]:<goal>. Available lifecycle phases are: validate, initialize, generate-sources, process-sources, generate-resources, process-resources, compile, process-classes, generate-test-sources, process-test-sources, generate-test-resources, process-test-resources, test-compile, process-test-classes, test, prepare-package, package, pre-integration-test, integration-test, post-integration-test, verify, install, deploy, pre-clean, clean, post-clean, pre-site, site, post-site, site-deploy. -> [Help 1]'), ['mvn clean package', 'mvn clean install']),
 | 
			
		||||
    (Command(script='mvn -N', stdout='[ERROR] No goals have been specified for this build. You must specify a valid lifecycle phase or a goal in the format <plugin-prefix>:<goal> or <plugin-group-id>:<plugin-artifact-id>[:<plugin-version>]:<goal>. Available lifecycle phases are: validate, initialize, generate-sources, process-sources, generate-resources, process-resources, compile, process-classes, generate-test-sources, process-test-sources, generate-test-resources, process-test-resources, test-compile, process-test-classes, test, prepare-package, package, pre-integration-test, integration-test, post-integration-test, verify, install, deploy, pre-clean, clean, post-clean, pre-site, site, post-site, site-deploy. -> [Help 1]'), ['mvn -N clean package', 'mvn -N clean install'])])
 | 
			
		||||
def test_get_new_command(command, new_command):
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -32,9 +32,9 @@ def test_match(command):
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command, new_command', [
 | 
			
		||||
    (Command(script='mvn cle', stdout='[ERROR] Unknown lifecycle phase "cle". You must specify a valid lifecycle phase or a goal in the format <plugin-prefix>:<goal> or <plugin-group-id>:<plugin-artifact-id>[:<plugin-version>]:<goal>. Available lifecycle phases are: validate, initialize, generate-sources, process-sources, generate-resources, process-resources, compile, process-classes, generate-test-sources, process-test-sources, generate-test-resources, process-test-resources, test-compile, process-test-classes, test, prepare-package, package, pre-integration-test, integration-test, post-integration-test, verify, install, deploy, pre-clean, clean, post-clean, pre-site, site, post-site, site-deploy. -> [Help 1]'), ['mvn clean', 'mvn compile']),
 | 
			
		||||
    (Command(script='mvn claen package', stdout='[ERROR] Unknown lifecycle phase "claen". You must specify a valid lifecycle phase or a goal in the format <plugin-prefix>:<goal> or <plugin-group-id>:<plugin-artifact-id>[:<plugin-version>]:<goal>. Available lifecycle phases are: validate, initialize, generate-sources, process-sources, generate-resources, process-resources, compile, process-classes, generate-test-sources, process-test-sources, generate-test-resources, process-test-resources, test-compile, process-test-classes, test, prepare-package, package, pre-integration-test, integration-test, post-integration-test, verify, install, deploy, pre-clean, clean, post-clean, pre-site, site, post-site, site-deploy. -> [Help 1]'), ['mvn clean package'])])
 | 
			
		||||
def test_get_new_command(command, new_command):
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -6,22 +6,37 @@ from tests.utils import Command
 | 
			
		||||
@pytest.fixture(autouse=True)
 | 
			
		||||
def get_all_executables(mocker):
 | 
			
		||||
    mocker.patch('thefuck.rules.no_command.get_all_executables',
 | 
			
		||||
                 return_value=['vim', 'apt-get', 'fsck'])
 | 
			
		||||
                 return_value=['vim', 'fsck', 'git', 'go'])
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture(autouse=True)
 | 
			
		||||
def history_without_current(mocker):
 | 
			
		||||
    return mocker.patch(
 | 
			
		||||
        'thefuck.rules.no_command.get_valid_history_without_current',
 | 
			
		||||
        return_value=['git commit'])
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('no_memoize')
 | 
			
		||||
def test_match():
 | 
			
		||||
    assert match(Command(stderr='vom: not found', script='vom file.py'))
 | 
			
		||||
    assert match(Command(stderr='fucck: not found', script='fucck'))
 | 
			
		||||
    assert not match(Command(stderr='qweqwe: not found', script='qweqwe'))
 | 
			
		||||
    assert not match(Command(stderr='some text', script='vom file.py'))
 | 
			
		||||
@pytest.mark.parametrize('script, stderr', [
 | 
			
		||||
    ('vom file.py', 'vom: not found'),
 | 
			
		||||
    ('fucck', 'fucck: not found'),
 | 
			
		||||
    ('got commit', 'got: command not found')])
 | 
			
		||||
def test_match(script, stderr):
 | 
			
		||||
    assert match(Command(script, stderr=stderr))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('no_memoize')
 | 
			
		||||
def test_get_new_command():
 | 
			
		||||
    assert get_new_command(
 | 
			
		||||
        Command(stderr='vom: not found',
 | 
			
		||||
                script='vom file.py')) == ['vim file.py']
 | 
			
		||||
    assert get_new_command(
 | 
			
		||||
        Command(stderr='fucck: not found',
 | 
			
		||||
                script='fucck')) == ['fsck']
 | 
			
		||||
@pytest.mark.parametrize('script, stderr', [
 | 
			
		||||
    ('qweqwe', 'qweqwe: not found'),
 | 
			
		||||
    ('vom file.py', 'some text')])
 | 
			
		||||
def test_not_match(script, stderr):
 | 
			
		||||
    assert not match(Command(script, stderr=stderr))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('no_memoize')
 | 
			
		||||
@pytest.mark.parametrize('script, result', [
 | 
			
		||||
    ('vom file.py', ['vim file.py']),
 | 
			
		||||
    ('fucck', ['fsck']),
 | 
			
		||||
    ('got commit', ['git commit', 'go commit'])])
 | 
			
		||||
def test_get_new_command(script, result):
 | 
			
		||||
    assert get_new_command(Command(script)) == result
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,7 @@ from tests.utils import Command
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='mv foo bar/foo', stderr="mv: cannot move 'foo' to 'bar/foo': No such file or directory"),
 | 
			
		||||
    Command(script='mv foo bar/', stderr="mv: cannot move 'foo' to 'bar/': No such file or directory"),
 | 
			
		||||
    ])
 | 
			
		||||
])
 | 
			
		||||
def test_match(command):
 | 
			
		||||
    assert match(command)
 | 
			
		||||
 | 
			
		||||
@@ -14,7 +14,7 @@ def test_match(command):
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='mv foo bar/', stderr=""),
 | 
			
		||||
    Command(script='mv foo bar/foo', stderr="mv: permission denied"),
 | 
			
		||||
    ])
 | 
			
		||||
])
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
@@ -22,6 +22,6 @@ def test_not_match(command):
 | 
			
		||||
@pytest.mark.parametrize('command, new_command', [
 | 
			
		||||
    (Command(script='mv foo bar/foo', stderr="mv: cannot move 'foo' to 'bar/foo': No such file or directory"), 'mkdir -p bar && mv foo bar/foo'),
 | 
			
		||||
    (Command(script='mv foo bar/', stderr="mv: cannot move 'foo' to 'bar/': No such file or directory"), 'mkdir -p bar && mv foo bar/'),
 | 
			
		||||
    ])
 | 
			
		||||
])
 | 
			
		||||
def test_get_new_command(command, new_command):
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										58
									
								
								tests/rules/test_npm_wrong_command.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								tests/rules/test_npm_wrong_command.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,58 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.npm_wrong_command import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
stdout = '''
 | 
			
		||||
Usage: npm <command>
 | 
			
		||||
 | 
			
		||||
where <command> is one of:
 | 
			
		||||
    access, add-user, adduser, apihelp, author, bin, bugs, c,
 | 
			
		||||
    cache, completion, config, ddp, dedupe, deprecate, dist-tag,
 | 
			
		||||
    dist-tags, docs, edit, explore, faq, find, find-dupes, get,
 | 
			
		||||
    help, help-search, home, i, info, init, install, issues, la,
 | 
			
		||||
    link, list, ll, ln, login, logout, ls, outdated, owner,
 | 
			
		||||
    pack, ping, prefix, prune, publish, r, rb, rebuild, remove,
 | 
			
		||||
    repo, restart, rm, root, run-script, s, se, search, set,
 | 
			
		||||
    show, shrinkwrap, star, stars, start, stop, t, tag, team,
 | 
			
		||||
    test, tst, un, uninstall, unlink, unpublish, unstar, up,
 | 
			
		||||
    update, upgrade, v, verison, version, view, whoami
 | 
			
		||||
 | 
			
		||||
npm <cmd> -h     quick help on <cmd>
 | 
			
		||||
npm -l           display full usage info
 | 
			
		||||
npm faq          commonly asked questions
 | 
			
		||||
npm help <term>  search for help on <term>
 | 
			
		||||
npm help npm     involved overview
 | 
			
		||||
 | 
			
		||||
Specify configs in the ini-formatted file:
 | 
			
		||||
    /home/nvbn/.npmrc
 | 
			
		||||
or on the command line via: npm <command> --key value
 | 
			
		||||
Config info can be viewed via: npm help config
 | 
			
		||||
 | 
			
		||||
npm@2.14.7 /opt/node/lib/node_modules/npm
 | 
			
		||||
'''
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script', [
 | 
			
		||||
    'npm urgrdae',
 | 
			
		||||
    'npm urgrade -g',
 | 
			
		||||
    'npm -f urgrade -g',
 | 
			
		||||
    'npm urg'])
 | 
			
		||||
def test_match(script):
 | 
			
		||||
    assert match(Command(script, stdout))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, stdout', [
 | 
			
		||||
    ('npm urgrade', ''),
 | 
			
		||||
    ('npm', stdout),
 | 
			
		||||
    ('test urgrade', stdout),
 | 
			
		||||
    ('npm -e', stdout)])
 | 
			
		||||
def test_not_match(script, stdout):
 | 
			
		||||
    assert not match(Command(script, stdout))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('script, result', [
 | 
			
		||||
    ('npm urgrade', 'npm upgrade'),
 | 
			
		||||
    ('npm -g isntall gulp', 'npm -g install gulp'),
 | 
			
		||||
    ('npm isntall -g gulp', 'npm install -g gulp')])
 | 
			
		||||
def test_get_new_command(script, result):
 | 
			
		||||
    assert get_new_command(Command(script, stdout)) == result
 | 
			
		||||
@@ -7,25 +7,25 @@ from tests.utils import Command
 | 
			
		||||
    Command('rm foo', stderr='rm: foo: is a directory'),
 | 
			
		||||
    Command('rm foo', stderr='rm: foo: Is a directory'),
 | 
			
		||||
    Command('hdfs dfs -rm foo', stderr='rm: `foo`: Is a directory'),
 | 
			
		||||
    Command('./bin/hdfs dfs -rm foo', stderr='rm: `foo`: Is a directory')
 | 
			
		||||
    ])
 | 
			
		||||
    Command('./bin/hdfs dfs -rm foo', stderr='rm: `foo`: Is a directory'),
 | 
			
		||||
])
 | 
			
		||||
def test_match(command):
 | 
			
		||||
    assert match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command('rm foo'), 
 | 
			
		||||
    Command('rm foo'),
 | 
			
		||||
    Command('hdfs dfs -rm foo'),
 | 
			
		||||
    Command('./bin/hdfs dfs -rm foo'),  
 | 
			
		||||
    Command()])
 | 
			
		||||
    Command('./bin/hdfs dfs -rm foo'),
 | 
			
		||||
    Command(),
 | 
			
		||||
])
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command, new_command', [
 | 
			
		||||
    (Command('rm foo'), 'rm -rf foo'),
 | 
			
		||||
    (Command('hdfs dfs -rm foo'), 'hdfs dfs -rm -r foo')])
 | 
			
		||||
    (Command('hdfs dfs -rm foo'), 'hdfs dfs -rm -r foo'),
 | 
			
		||||
])
 | 
			
		||||
def test_get_new_command(command, new_command):
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import os
 | 
			
		||||
import pytest
 | 
			
		||||
from mock import Mock
 | 
			
		||||
from thefuck.rules.ssh_known_hosts import match, get_new_command,\
 | 
			
		||||
    side_effect
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
@@ -53,6 +52,7 @@ def test_match(ssh_error):
 | 
			
		||||
    assert not match(Command('ssh'))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.skipif(os.name == 'nt', reason='Skip if testing on Windows')
 | 
			
		||||
def test_side_effect(ssh_error):
 | 
			
		||||
    errormsg, path, reset, known_hosts = ssh_error
 | 
			
		||||
    command = Command('ssh user@host', stderr=errormsg)
 | 
			
		||||
 
 | 
			
		||||
@@ -19,11 +19,13 @@ def test_match(stderr, stdout):
 | 
			
		||||
 | 
			
		||||
def test_not_match():
 | 
			
		||||
    assert not match(Command())
 | 
			
		||||
    assert not match(Command(script='sudo ls', stderr='Permission denied'))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('before, after', [
 | 
			
		||||
    ('ls', 'sudo ls'),
 | 
			
		||||
    ('echo a > b', 'sudo sh -c "echo a > b"'),
 | 
			
		||||
    ('echo "a" >> b', 'sudo sh -c "echo \\"a\\" >> b"')])
 | 
			
		||||
    ('echo "a" >> b', 'sudo sh -c "echo \\"a\\" >> b"'),
 | 
			
		||||
    ('mkdir && touch a', 'sudo sh -c "mkdir && touch a"')])
 | 
			
		||||
def test_get_new_command(before, after):
 | 
			
		||||
    assert get_new_command(Command(before)) == after
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,3 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.rules.systemctl import match, get_new_command
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -14,8 +14,8 @@ def test_match(command):
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='./bin/hdfs dfs -ls', stderr=''),
 | 
			
		||||
    Command(script='./bin/hdfs dfs -ls /foo/bar', stderr=''),  
 | 
			
		||||
    Command(script='hdfs dfs -ls -R /foo/bar', stderr=''),  
 | 
			
		||||
    Command(script='./bin/hdfs dfs -ls /foo/bar', stderr=''),
 | 
			
		||||
    Command(script='hdfs dfs -ls -R /foo/bar', stderr=''),
 | 
			
		||||
    Command()])
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
@@ -32,4 +32,3 @@ def test_not_match(command):
 | 
			
		||||
        stderr='ls: Unknown command\nDid you mean -ls?  This command begins with a dash.'), ['./bin/hdfs dfs -Dtest=fred -ls -R /foo/bar'])])
 | 
			
		||||
def test_get_new_command(command, new_command):
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -16,8 +16,8 @@ def test_match(command):
 | 
			
		||||
 | 
			
		||||
@pytest.mark.parametrize('command', [
 | 
			
		||||
    Command(script='vagrant ssh', stderr=''),
 | 
			
		||||
    Command(script='vagrant ssh jeff', stderr='The machine with the name \'jeff\' was not found configured for this Vagrant environment.'),  
 | 
			
		||||
    Command(script='vagrant ssh', stderr='A Vagrant environment or target machine is required to run this command. Run `vagrant init` to create a new Vagrant environment. Or, get an ID of a target machine from `vagrant global-status` to run this command on. A final option is to change to a directory with a Vagrantfile and to try again.'),  
 | 
			
		||||
    Command(script='vagrant ssh jeff', stderr='The machine with the name \'jeff\' was not found configured for this Vagrant environment.'),
 | 
			
		||||
    Command(script='vagrant ssh', stderr='A Vagrant environment or target machine is required to run this command. Run `vagrant init` to create a new Vagrant environment. Or, get an ID of a target machine from `vagrant global-status` to run this command on. A final option is to change to a directory with a Vagrantfile and to try again.'),
 | 
			
		||||
    Command()])
 | 
			
		||||
def test_not_match(command):
 | 
			
		||||
    assert not match(command)
 | 
			
		||||
@@ -32,4 +32,3 @@ def test_not_match(command):
 | 
			
		||||
            stderr='VM must be created before running this command. Run `vagrant up` first.'), ['vagrant up devbox && vagrant rdp devbox', 'vagrant up && vagrant rdp devbox'])])
 | 
			
		||||
def test_get_new_command(command, new_command):
 | 
			
		||||
    assert get_new_command(command) == new_command
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										0
									
								
								tests/shells/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								tests/shells/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										22
									
								
								tests/shells/conftest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								tests/shells/conftest.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,22 @@
 | 
			
		||||
import pytest
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def builtins_open(mocker):
 | 
			
		||||
    return mocker.patch('six.moves.builtins.open')
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def isfile(mocker):
 | 
			
		||||
    return mocker.patch('os.path.isfile', return_value=True)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
@pytest.mark.usefixtures('isfile')
 | 
			
		||||
def history_lines(mocker):
 | 
			
		||||
    def aux(lines):
 | 
			
		||||
        mock = mocker.patch('io.open')
 | 
			
		||||
        mock.return_value.__enter__ \
 | 
			
		||||
            .return_value.readlines.return_value = lines
 | 
			
		||||
 | 
			
		||||
    return aux
 | 
			
		||||
							
								
								
									
										58
									
								
								tests/shells/test_bash.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								tests/shells/test_bash.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,58 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import os
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.shells import Bash
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('isfile', 'no_memoize', 'no_cache')
 | 
			
		||||
class TestBash(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return Bash()
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def shell_aliases(self):
 | 
			
		||||
        os.environ['TF_SHELL_ALIASES'] = (
 | 
			
		||||
            'alias fuck=\'eval $(thefuck $(fc -ln -1))\'\n'
 | 
			
		||||
            'alias l=\'ls -CF\'\n'
 | 
			
		||||
            'alias la=\'ls -A\'\n'
 | 
			
		||||
            'alias ll=\'ls -alF\'')
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('before, after', [
 | 
			
		||||
        ('pwd', 'pwd'),
 | 
			
		||||
        ('fuck', 'eval $(thefuck $(fc -ln -1))'),
 | 
			
		||||
        ('awk', 'awk'),
 | 
			
		||||
        ('ll', 'ls -alF')])
 | 
			
		||||
    def test_from_shell(self, before, after, shell):
 | 
			
		||||
        assert shell.from_shell(before) == after
 | 
			
		||||
 | 
			
		||||
    def test_to_shell(self, shell):
 | 
			
		||||
        assert shell.to_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('ls', 'cd') == 'ls && cd'
 | 
			
		||||
 | 
			
		||||
    def test_get_aliases(self, shell):
 | 
			
		||||
        assert shell.get_aliases() == {'fuck': 'eval $(thefuck $(fc -ln -1))',
 | 
			
		||||
                                       'l': 'ls -CF',
 | 
			
		||||
                                       'la': 'ls -A',
 | 
			
		||||
                                       'll': 'ls -alF'}
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'alias fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'alias FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'TF_ALIAS=fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'PYTHONIOENCODING=utf-8' in shell.app_alias('fuck')
 | 
			
		||||
 | 
			
		||||
    def test_app_alias_variables_correctly_set(self, shell):
 | 
			
		||||
        alias = shell.app_alias('fuck')
 | 
			
		||||
        assert "alias fuck='TF_CMD=$(TF_ALIAS" in alias
 | 
			
		||||
        assert '$(TF_ALIAS=fuck PYTHONIOENCODING' in alias
 | 
			
		||||
        assert 'PYTHONIOENCODING=utf-8 TF_SHELL_ALIASES' in alias
 | 
			
		||||
        assert 'ALIASES=$(alias) thefuck' in alias
 | 
			
		||||
 | 
			
		||||
    def test_get_history(self, history_lines, shell):
 | 
			
		||||
        history_lines(['ls', 'rm'])
 | 
			
		||||
        assert list(shell.get_history()) == ['ls', 'rm']
 | 
			
		||||
							
								
								
									
										80
									
								
								tests/shells/test_fish.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								tests/shells/test_fish.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,80 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.shells import Fish
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('isfile', 'no_memoize', 'no_cache')
 | 
			
		||||
class TestFish(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return Fish()
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def Popen(self, mocker):
 | 
			
		||||
        mock = mocker.patch('thefuck.shells.fish.Popen')
 | 
			
		||||
        mock.return_value.stdout.read.return_value = (
 | 
			
		||||
            b'cd\nfish_config\nfuck\nfunced\nfuncsave\ngrep\nhistory\nll\nls\n'
 | 
			
		||||
            b'man\nmath\npopd\npushd\nruby')
 | 
			
		||||
        return mock
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def os_environ(self, monkeypatch, key, value):
 | 
			
		||||
        monkeypatch.setattr('os.environ', {key: value})
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('key, value', [
 | 
			
		||||
        ('TF_OVERRIDDEN_ALIASES', 'cut,git,sed'),  # legacy
 | 
			
		||||
        ('THEFUCK_OVERRIDDEN_ALIASES', 'cut,git,sed'),
 | 
			
		||||
        ('THEFUCK_OVERRIDDEN_ALIASES', 'cut, git, sed'),
 | 
			
		||||
        ('THEFUCK_OVERRIDDEN_ALIASES', ' cut,\tgit,sed\n'),
 | 
			
		||||
        ('THEFUCK_OVERRIDDEN_ALIASES', '\ncut,\n\ngit,\tsed\r')])
 | 
			
		||||
    def test_get_overridden_aliases(self, shell, os_environ):
 | 
			
		||||
        assert shell._get_overridden_aliases() == {'cd', 'cut', 'git', 'grep',
 | 
			
		||||
                                                   'ls', 'man', 'open', 'sed'}
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('before, after', [
 | 
			
		||||
        ('cd', 'cd'),
 | 
			
		||||
        ('pwd', 'pwd'),
 | 
			
		||||
        ('fuck', 'fish -ic "fuck"'),
 | 
			
		||||
        ('find', 'find'),
 | 
			
		||||
        ('funced', 'fish -ic "funced"'),
 | 
			
		||||
        ('grep', 'grep'),
 | 
			
		||||
        ('awk', 'awk'),
 | 
			
		||||
        ('math "2 + 2"', r'fish -ic "math \"2 + 2\""'),
 | 
			
		||||
        ('man', 'man'),
 | 
			
		||||
        ('open', 'open'),
 | 
			
		||||
        ('vim', 'vim'),
 | 
			
		||||
        ('ll', 'fish -ic "ll"'),
 | 
			
		||||
        ('ls', 'ls')])  # Fish has no aliases but functions
 | 
			
		||||
    def test_from_shell(self, before, after, shell):
 | 
			
		||||
        assert shell.from_shell(before) == after
 | 
			
		||||
 | 
			
		||||
    def test_to_shell(self, shell):
 | 
			
		||||
        assert shell.to_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('foo', 'bar') == 'foo; and bar'
 | 
			
		||||
 | 
			
		||||
    def test_get_aliases(self, shell):
 | 
			
		||||
        assert shell.get_aliases() == {'fish_config': 'fish_config',
 | 
			
		||||
                                       'fuck': 'fuck',
 | 
			
		||||
                                       'funced': 'funced',
 | 
			
		||||
                                       'funcsave': 'funcsave',
 | 
			
		||||
                                       'history': 'history',
 | 
			
		||||
                                       'll': 'll',
 | 
			
		||||
                                       'math': 'math',
 | 
			
		||||
                                       'popd': 'popd',
 | 
			
		||||
                                       'pushd': 'pushd',
 | 
			
		||||
                                       'ruby': 'ruby'}
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'function fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'function FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'TF_ALIAS=fuck PYTHONIOENCODING' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'PYTHONIOENCODING=utf-8 thefuck' in shell.app_alias('fuck')
 | 
			
		||||
 | 
			
		||||
    def test_get_history(self, history_lines, shell):
 | 
			
		||||
        history_lines(['- cmd: ls', '  when: 1432613911',
 | 
			
		||||
                       '- cmd: rm', '  when: 1432613916'])
 | 
			
		||||
        assert list(shell.get_history()) == ['ls', 'rm']
 | 
			
		||||
							
								
								
									
										39
									
								
								tests/shells/test_generic.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								tests/shells/test_generic.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.shells import Generic
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TestGeneric(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return Generic()
 | 
			
		||||
 | 
			
		||||
    def test_from_shell(self, shell):
 | 
			
		||||
        assert shell.from_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_to_shell(self, shell):
 | 
			
		||||
        assert shell.to_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('ls', 'cd') == 'ls && cd'
 | 
			
		||||
 | 
			
		||||
    def test_get_aliases(self, shell):
 | 
			
		||||
        assert shell.get_aliases() == {}
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'alias fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'alias FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'TF_ALIAS=fuck PYTHONIOENCODING' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'PYTHONIOENCODING=utf-8 thefuck' in shell.app_alias('fuck')
 | 
			
		||||
 | 
			
		||||
    def test_get_history(self, history_lines, shell):
 | 
			
		||||
        history_lines(['ls', 'rm'])
 | 
			
		||||
        # We don't know what to do in generic shell with history lines,
 | 
			
		||||
        # so just ignore them:
 | 
			
		||||
        assert list(shell.get_history()) == []
 | 
			
		||||
 | 
			
		||||
    def test_split_command(self, shell):
 | 
			
		||||
        assert shell.split_command('ls') == ['ls']
 | 
			
		||||
        assert shell.split_command(u'echo café') == [u'echo', u'café']
 | 
			
		||||
							
								
								
									
										19
									
								
								tests/shells/test_powershell.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								tests/shells/test_powershell.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.shells import Powershell
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('isfile', 'no_memoize', 'no_cache')
 | 
			
		||||
class TestPowershell(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return Powershell()
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('ls', 'cd') == '(ls) -and (cd)'
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'function fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'function FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
							
								
								
									
										50
									
								
								tests/shells/test_tcsh.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								tests/shells/test_tcsh.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,50 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.shells.tcsh import Tcsh
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('isfile', 'no_memoize', 'no_cache')
 | 
			
		||||
class TestTcsh(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return Tcsh()
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def Popen(self, mocker):
 | 
			
		||||
        mock = mocker.patch('thefuck.shells.tcsh.Popen')
 | 
			
		||||
        mock.return_value.stdout.read.return_value = (
 | 
			
		||||
            b'fuck\teval $(thefuck $(fc -ln -1))\n'
 | 
			
		||||
            b'l\tls -CF\n'
 | 
			
		||||
            b'la\tls -A\n'
 | 
			
		||||
            b'll\tls -alF')
 | 
			
		||||
        return mock
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('before, after', [
 | 
			
		||||
        ('pwd', 'pwd'),
 | 
			
		||||
        ('fuck', 'eval $(thefuck $(fc -ln -1))'),
 | 
			
		||||
        ('awk', 'awk'),
 | 
			
		||||
        ('ll', 'ls -alF')])
 | 
			
		||||
    def test_from_shell(self, before, after, shell):
 | 
			
		||||
        assert shell.from_shell(before) == after
 | 
			
		||||
 | 
			
		||||
    def test_to_shell(self, shell):
 | 
			
		||||
        assert shell.to_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('ls', 'cd') == 'ls && cd'
 | 
			
		||||
 | 
			
		||||
    def test_get_aliases(self, shell):
 | 
			
		||||
        assert shell.get_aliases() == {'fuck': 'eval $(thefuck $(fc -ln -1))',
 | 
			
		||||
                                       'l': 'ls -CF',
 | 
			
		||||
                                       'la': 'ls -A',
 | 
			
		||||
                                       'll': 'ls -alF'}
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'alias fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'alias FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
 | 
			
		||||
    def test_get_history(self, history_lines, shell):
 | 
			
		||||
        history_lines(['ls', 'rm'])
 | 
			
		||||
        assert list(shell.get_history()) == ['ls', 'rm']
 | 
			
		||||
							
								
								
									
										57
									
								
								tests/shells/test_zsh.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								tests/shells/test_zsh.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,57 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import os
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck.shells.zsh import Zsh
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('isfile', 'no_memoize', 'no_cache')
 | 
			
		||||
class TestZsh(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return Zsh()
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def shell_aliases(self):
 | 
			
		||||
        os.environ['TF_SHELL_ALIASES'] = (
 | 
			
		||||
            'fuck=\'eval $(thefuck $(fc -ln -1 | tail -n 1))\'\n'
 | 
			
		||||
            'l=\'ls -CF\'\n'
 | 
			
		||||
            'la=\'ls -A\'\n'
 | 
			
		||||
            'll=\'ls -alF\'')
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('before, after', [
 | 
			
		||||
        ('fuck', 'eval $(thefuck $(fc -ln -1 | tail -n 1))'),
 | 
			
		||||
        ('pwd', 'pwd'),
 | 
			
		||||
        ('ll', 'ls -alF')])
 | 
			
		||||
    def test_from_shell(self, before, after, shell):
 | 
			
		||||
        assert shell.from_shell(before) == after
 | 
			
		||||
 | 
			
		||||
    def test_to_shell(self, shell):
 | 
			
		||||
        assert shell.to_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('ls', 'cd') == 'ls && cd'
 | 
			
		||||
 | 
			
		||||
    def test_get_aliases(self, shell):
 | 
			
		||||
        assert shell.get_aliases() == {
 | 
			
		||||
            'fuck': 'eval $(thefuck $(fc -ln -1 | tail -n 1))',
 | 
			
		||||
            'l': 'ls -CF',
 | 
			
		||||
            'la': 'ls -A',
 | 
			
		||||
            'll': 'ls -alF'}
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'alias fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'alias FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'PYTHONIOENCODING' in shell.app_alias('fuck')
 | 
			
		||||
 | 
			
		||||
    def test_app_alias_variables_correctly_set(self, shell):
 | 
			
		||||
        alias = shell.app_alias('fuck')
 | 
			
		||||
        assert "alias fuck='TF_CMD=$(TF_ALIAS" in alias
 | 
			
		||||
        assert '$(TF_ALIAS=fuck PYTHONIOENCODING' in alias
 | 
			
		||||
        assert 'PYTHONIOENCODING=utf-8 TF_SHELL_ALIASES' in alias
 | 
			
		||||
        assert 'ALIASES=$(alias) thefuck' in alias
 | 
			
		||||
 | 
			
		||||
    def test_get_history(self, history_lines, shell):
 | 
			
		||||
        history_lines([': 1432613911:0;ls', ': 1432613916:0;rm'])
 | 
			
		||||
        assert list(shell.get_history()) == ['ls', 'rm']
 | 
			
		||||
@@ -1,5 +1,4 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from mock import Mock
 | 
			
		||||
from thefuck.specific.sudo import sudo_support
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,7 @@
 | 
			
		||||
import pytest
 | 
			
		||||
import six
 | 
			
		||||
from mock import Mock
 | 
			
		||||
from thefuck import conf
 | 
			
		||||
from thefuck import const
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
@@ -20,7 +20,7 @@ def environ(monkeypatch):
 | 
			
		||||
def test_settings_defaults(load_source, settings):
 | 
			
		||||
    load_source.return_value = object()
 | 
			
		||||
    settings.init()
 | 
			
		||||
    for key, val in conf.DEFAULT_SETTINGS.items():
 | 
			
		||||
    for key, val in const.DEFAULT_SETTINGS.items():
 | 
			
		||||
        assert getattr(settings, key) == val
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -42,13 +42,13 @@ class TestSettingsFromFile(object):
 | 
			
		||||
        assert settings.exclude_rules == ['git']
 | 
			
		||||
 | 
			
		||||
    def test_from_file_with_DEFAULT(self, load_source, settings):
 | 
			
		||||
        load_source.return_value = Mock(rules=conf.DEFAULT_RULES + ['test'],
 | 
			
		||||
        load_source.return_value = Mock(rules=const.DEFAULT_RULES + ['test'],
 | 
			
		||||
                                        wait_command=10,
 | 
			
		||||
                                        exclude_rules=[],
 | 
			
		||||
                                        require_confirmation=True,
 | 
			
		||||
                                        no_colors=True)
 | 
			
		||||
        settings.init()
 | 
			
		||||
        assert settings.rules == conf.DEFAULT_RULES + ['test']
 | 
			
		||||
        assert settings.rules == const.DEFAULT_RULES + ['test']
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixture('load_source')
 | 
			
		||||
@@ -71,7 +71,7 @@ class TestSettingsFromEnv(object):
 | 
			
		||||
    def test_from_env_with_DEFAULT(self, environ, settings):
 | 
			
		||||
        environ.update({'THEFUCK_RULES': 'DEFAULT_RULES:bash:lisp'})
 | 
			
		||||
        settings.init()
 | 
			
		||||
        assert settings.rules == conf.DEFAULT_RULES + ['bash', 'lisp']
 | 
			
		||||
        assert settings.rules == const.DEFAULT_RULES + ['bash', 'lisp']
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TestInitializeSettingsFile(object):
 | 
			
		||||
@@ -93,7 +93,7 @@ class TestInitializeSettingsFile(object):
 | 
			
		||||
        settings_file_contents = settings_file.getvalue()
 | 
			
		||||
        assert settings_path_mock.is_file.call_count == 1
 | 
			
		||||
        assert settings_path_mock.open.call_count == 1
 | 
			
		||||
        assert conf.SETTINGS_HEADER in settings_file_contents
 | 
			
		||||
        for setting in conf.DEFAULT_SETTINGS.items():
 | 
			
		||||
        assert const.SETTINGS_HEADER in settings_file_contents
 | 
			
		||||
        for setting in const.DEFAULT_SETTINGS.items():
 | 
			
		||||
            assert '# {} = {}\n'.format(*setting) in settings_file_contents
 | 
			
		||||
        settings_file.close()
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,8 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import pytest
 | 
			
		||||
from pathlib import PosixPath
 | 
			
		||||
from thefuck import corrector, conf
 | 
			
		||||
from pathlib import Path
 | 
			
		||||
from thefuck import corrector, const
 | 
			
		||||
from tests.utils import Rule, Command, CorrectedCommand
 | 
			
		||||
from thefuck.corrector import get_corrected_commands, organize_commands
 | 
			
		||||
 | 
			
		||||
@@ -22,13 +24,13 @@ class TestGetRules(object):
 | 
			
		||||
        assert {r.name for r in rules} == set(names)
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('paths, conf_rules, exclude_rules, loaded_rules', [
 | 
			
		||||
        (['git.py', 'bash.py'], conf.DEFAULT_RULES, [], ['git', 'bash']),
 | 
			
		||||
        (['git.py', 'bash.py'], const.DEFAULT_RULES, [], ['git', 'bash']),
 | 
			
		||||
        (['git.py', 'bash.py'], ['git'], [], ['git']),
 | 
			
		||||
        (['git.py', 'bash.py'], conf.DEFAULT_RULES, ['git'], ['bash']),
 | 
			
		||||
        (['git.py', 'bash.py'], const.DEFAULT_RULES, ['git'], ['bash']),
 | 
			
		||||
        (['git.py', 'bash.py'], ['git'], ['git'], [])])
 | 
			
		||||
    def test_get_rules(self, glob, settings, paths, conf_rules, exclude_rules,
 | 
			
		||||
                       loaded_rules):
 | 
			
		||||
        glob([PosixPath(path) for path in paths])
 | 
			
		||||
        glob([Path(path) for path in paths])
 | 
			
		||||
        settings.update(rules=conf_rules,
 | 
			
		||||
                        priority={},
 | 
			
		||||
                        exclude_rules=exclude_rules)
 | 
			
		||||
@@ -53,7 +55,9 @@ def test_organize_commands():
 | 
			
		||||
    """Ensures that the function removes duplicates and sorts commands."""
 | 
			
		||||
    commands = [CorrectedCommand('ls'), CorrectedCommand('ls -la', priority=9000),
 | 
			
		||||
                CorrectedCommand('ls -lh', priority=100),
 | 
			
		||||
                CorrectedCommand(u'echo café', priority=200),
 | 
			
		||||
                CorrectedCommand('ls -lh', priority=9999)]
 | 
			
		||||
    assert list(organize_commands(iter(commands))) \
 | 
			
		||||
        == [CorrectedCommand('ls'), CorrectedCommand('ls -lh', priority=100),
 | 
			
		||||
            CorrectedCommand(u'echo café', priority=200),
 | 
			
		||||
            CorrectedCommand('ls -la', priority=9000)]
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,4 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from mock import Mock
 | 
			
		||||
from thefuck import logs
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,235 +0,0 @@
 | 
			
		||||
import pytest
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def builtins_open(mocker):
 | 
			
		||||
    return mocker.patch('six.moves.builtins.open')
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def isfile(mocker):
 | 
			
		||||
    return mocker.patch('os.path.isfile', return_value=True)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
@pytest.mark.usefixtures('isfile')
 | 
			
		||||
def history_lines(mocker):
 | 
			
		||||
    def aux(lines):
 | 
			
		||||
        mock = mocker.patch('io.open')
 | 
			
		||||
        mock.return_value.__enter__\
 | 
			
		||||
            .return_value.__iter__.return_value = lines
 | 
			
		||||
    return aux
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TestGeneric(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return shells.Generic()
 | 
			
		||||
 | 
			
		||||
    def test_from_shell(self, shell):
 | 
			
		||||
        assert shell.from_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_to_shell(self, shell):
 | 
			
		||||
        assert shell.to_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_put_to_history(self, builtins_open, shell):
 | 
			
		||||
        assert shell.put_to_history('ls') is None
 | 
			
		||||
        assert builtins_open.call_count == 0
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('ls', 'cd') == 'ls && cd'
 | 
			
		||||
 | 
			
		||||
    def test_get_aliases(self, shell):
 | 
			
		||||
        assert shell.get_aliases() == {}
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'alias fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'alias FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'TF_ALIAS' in shell.app_alias('fuck')
 | 
			
		||||
 | 
			
		||||
    def test_get_history(self, history_lines, shell):
 | 
			
		||||
        history_lines(['ls', 'rm'])
 | 
			
		||||
        # We don't know what to do in generic shell with history lines,
 | 
			
		||||
        # so just ignore them:
 | 
			
		||||
        assert list(shell.get_history()) == []
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('isfile')
 | 
			
		||||
class TestBash(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return shells.Bash()
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def Popen(self, mocker):
 | 
			
		||||
        mock = mocker.patch('thefuck.shells.Popen')
 | 
			
		||||
        mock.return_value.stdout.read.return_value = (
 | 
			
		||||
            b'alias fuck=\'eval $(thefuck $(fc -ln -1))\'\n'
 | 
			
		||||
            b'alias l=\'ls -CF\'\n'
 | 
			
		||||
            b'alias la=\'ls -A\'\n'
 | 
			
		||||
            b'alias ll=\'ls -alF\'')
 | 
			
		||||
        return mock
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('before, after', [
 | 
			
		||||
        ('pwd', 'pwd'),
 | 
			
		||||
        ('fuck', 'eval $(thefuck $(fc -ln -1))'),
 | 
			
		||||
        ('awk', 'awk'),
 | 
			
		||||
        ('ll', 'ls -alF')])
 | 
			
		||||
    def test_from_shell(self, before, after, shell):
 | 
			
		||||
        assert shell.from_shell(before) == after
 | 
			
		||||
 | 
			
		||||
    def test_to_shell(self, shell):
 | 
			
		||||
        assert shell.to_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_put_to_history(self, builtins_open, shell):
 | 
			
		||||
        shell.put_to_history('ls')
 | 
			
		||||
        builtins_open.return_value.__enter__.return_value. \
 | 
			
		||||
            write.assert_called_once_with('ls\n')
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('ls', 'cd') == 'ls && cd'
 | 
			
		||||
 | 
			
		||||
    def test_get_aliases(self, shell):
 | 
			
		||||
        assert shell.get_aliases() == {'fuck': 'eval $(thefuck $(fc -ln -1))',
 | 
			
		||||
                                       'l': 'ls -CF',
 | 
			
		||||
                                       'la': 'ls -A',
 | 
			
		||||
                                       'll': 'ls -alF'}
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'alias fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'alias FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'TF_ALIAS' in shell.app_alias('fuck')
 | 
			
		||||
 | 
			
		||||
    def test_get_history(self, history_lines, shell):
 | 
			
		||||
        history_lines(['ls', 'rm'])
 | 
			
		||||
        assert list(shell.get_history()) == ['ls', 'rm']
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('isfile')
 | 
			
		||||
class TestFish(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return shells.Fish()
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def Popen(self, mocker):
 | 
			
		||||
        mock = mocker.patch('thefuck.shells.Popen')
 | 
			
		||||
        mock.return_value.stdout.read.return_value = (
 | 
			
		||||
            b'cd\nfish_config\nfuck\nfunced\nfuncsave\ngrep\nhistory\nll\nls\n'
 | 
			
		||||
            b'man\nmath\npopd\npushd\nruby')
 | 
			
		||||
        return mock
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def environ(self, monkeypatch):
 | 
			
		||||
        data = {'TF_OVERRIDDEN_ALIASES': 'cd, ls, man, open'}
 | 
			
		||||
        monkeypatch.setattr('thefuck.shells.os.environ', data)
 | 
			
		||||
        return data
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.usefixture('environ')
 | 
			
		||||
    def test_get_overridden_aliases(self, shell, environ):
 | 
			
		||||
        assert shell._get_overridden_aliases() == ['cd', 'ls', 'man', 'open']
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('before, after', [
 | 
			
		||||
        ('cd', 'cd'),
 | 
			
		||||
        ('pwd', 'pwd'),
 | 
			
		||||
        ('fuck', 'fish -ic "fuck"'),
 | 
			
		||||
        ('find', 'find'),
 | 
			
		||||
        ('funced', 'fish -ic "funced"'),
 | 
			
		||||
        ('grep', 'grep'),
 | 
			
		||||
        ('awk', 'awk'),
 | 
			
		||||
        ('math "2 + 2"', r'fish -ic "math \"2 + 2\""'),
 | 
			
		||||
        ('man', 'man'),
 | 
			
		||||
        ('open', 'open'),
 | 
			
		||||
        ('vim', 'vim'),
 | 
			
		||||
        ('ll', 'fish -ic "ll"'),
 | 
			
		||||
        ('ls', 'ls')])  # Fish has no aliases but functions
 | 
			
		||||
    def test_from_shell(self, before, after, shell):
 | 
			
		||||
        assert shell.from_shell(before) == after
 | 
			
		||||
 | 
			
		||||
    def test_to_shell(self, shell):
 | 
			
		||||
        assert shell.to_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_put_to_history(self, builtins_open, mocker, shell):
 | 
			
		||||
        mocker.patch('thefuck.shells.time',
 | 
			
		||||
                     return_value=1430707243.3517463)
 | 
			
		||||
        shell.put_to_history('ls')
 | 
			
		||||
        builtins_open.return_value.__enter__.return_value. \
 | 
			
		||||
            write.assert_called_once_with('- cmd: ls\n   when: 1430707243\n')
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('foo', 'bar') == 'foo; and bar'
 | 
			
		||||
 | 
			
		||||
    def test_get_aliases(self, shell):
 | 
			
		||||
        assert shell.get_aliases() == {'fish_config': 'fish_config',
 | 
			
		||||
                                       'fuck': 'fuck',
 | 
			
		||||
                                       'funced': 'funced',
 | 
			
		||||
                                       'funcsave': 'funcsave',
 | 
			
		||||
                                       'history': 'history',
 | 
			
		||||
                                       'll': 'll',
 | 
			
		||||
                                       'math': 'math',
 | 
			
		||||
                                       'popd': 'popd',
 | 
			
		||||
                                       'pushd': 'pushd',
 | 
			
		||||
                                       'ruby': 'ruby'}
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'function fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'function FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'TF_ALIAS' in shell.app_alias('fuck')
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.mark.usefixtures('isfile')
 | 
			
		||||
class TestZsh(object):
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def shell(self):
 | 
			
		||||
        return shells.Zsh()
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def Popen(self, mocker):
 | 
			
		||||
        mock = mocker.patch('thefuck.shells.Popen')
 | 
			
		||||
        mock.return_value.stdout.read.return_value = (
 | 
			
		||||
            b'fuck=\'eval $(thefuck $(fc -ln -1 | tail -n 1))\'\n'
 | 
			
		||||
            b'l=\'ls -CF\'\n'
 | 
			
		||||
            b'la=\'ls -A\'\n'
 | 
			
		||||
            b'll=\'ls -alF\'')
 | 
			
		||||
        return mock
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('before, after', [
 | 
			
		||||
        ('fuck', 'eval $(thefuck $(fc -ln -1 | tail -n 1))'),
 | 
			
		||||
        ('pwd', 'pwd'),
 | 
			
		||||
        ('ll', 'ls -alF')])
 | 
			
		||||
    def test_from_shell(self, before, after, shell):
 | 
			
		||||
        assert shell.from_shell(before) == after
 | 
			
		||||
 | 
			
		||||
    def test_to_shell(self, shell):
 | 
			
		||||
        assert shell.to_shell('pwd') == 'pwd'
 | 
			
		||||
 | 
			
		||||
    def test_put_to_history(self, builtins_open, mocker, shell):
 | 
			
		||||
        mocker.patch('thefuck.shells.time',
 | 
			
		||||
                     return_value=1430707243.3517463)
 | 
			
		||||
        shell.put_to_history('ls')
 | 
			
		||||
        builtins_open.return_value.__enter__.return_value. \
 | 
			
		||||
            write.assert_called_once_with(': 1430707243:0;ls\n')
 | 
			
		||||
 | 
			
		||||
    def test_and_(self, shell):
 | 
			
		||||
        assert shell.and_('ls', 'cd') == 'ls && cd'
 | 
			
		||||
 | 
			
		||||
    def test_get_aliases(self, shell):
 | 
			
		||||
        assert shell.get_aliases() == {
 | 
			
		||||
            'fuck': 'eval $(thefuck $(fc -ln -1 | tail -n 1))',
 | 
			
		||||
            'l': 'ls -CF',
 | 
			
		||||
            'la': 'ls -A',
 | 
			
		||||
            'll': 'ls -alF'}
 | 
			
		||||
 | 
			
		||||
    def test_app_alias(self, shell):
 | 
			
		||||
        assert 'alias fuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'alias FUCK' in shell.app_alias('FUCK')
 | 
			
		||||
        assert 'thefuck' in shell.app_alias('fuck')
 | 
			
		||||
        assert 'TF_ALIAS' in shell.app_alias('fuck')
 | 
			
		||||
 | 
			
		||||
    def test_get_history(self, history_lines, shell):
 | 
			
		||||
        history_lines([': 1432613911:0;ls', ': 1432613916:0;rm'])
 | 
			
		||||
        assert list(shell.get_history()) == ['ls', 'rm']
 | 
			
		||||
@@ -1,9 +1,12 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
import os
 | 
			
		||||
from subprocess import PIPE
 | 
			
		||||
from mock import Mock
 | 
			
		||||
from pathlib import Path
 | 
			
		||||
import pytest
 | 
			
		||||
from tests.utils import CorrectedCommand, Rule, Command
 | 
			
		||||
from thefuck import conf
 | 
			
		||||
from thefuck import const
 | 
			
		||||
from thefuck.exceptions import EmptyCommand
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -19,6 +22,12 @@ class TestCorrectedCommand(object):
 | 
			
		||||
        assert {CorrectedCommand('ls', None, 100),
 | 
			
		||||
                CorrectedCommand('ls', None, 200)} == {CorrectedCommand('ls')}
 | 
			
		||||
 | 
			
		||||
    def test_representable(self):
 | 
			
		||||
        assert '{}'.format(CorrectedCommand('ls', None, 100)) == \
 | 
			
		||||
               'CorrectedCommand(script=ls, side_effect=None, priority=100)'
 | 
			
		||||
        assert u'{}'.format(CorrectedCommand(u'echo café', None, 100)) == \
 | 
			
		||||
               u'CorrectedCommand(script=echo café, side_effect=None, priority=100)'
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TestRule(object):
 | 
			
		||||
    def test_from_path(self, mocker):
 | 
			
		||||
@@ -31,19 +40,20 @@ class TestRule(object):
 | 
			
		||||
                              enabled_by_default=True,
 | 
			
		||||
                              priority=900,
 | 
			
		||||
                              requires_output=True))
 | 
			
		||||
        assert Rule.from_path(Path('/rules/bash.py')) \
 | 
			
		||||
        rule_path = os.path.join(os.sep, 'rules', 'bash.py')
 | 
			
		||||
        assert Rule.from_path(Path(rule_path)) \
 | 
			
		||||
               == Rule('bash', match, get_new_command, priority=900)
 | 
			
		||||
        load_source.assert_called_once_with('bash', '/rules/bash.py')
 | 
			
		||||
        load_source.assert_called_once_with('bash', rule_path)
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('rules, exclude_rules, rule, is_enabled', [
 | 
			
		||||
        (conf.DEFAULT_RULES, [], Rule('git', enabled_by_default=True), True),
 | 
			
		||||
        (conf.DEFAULT_RULES, [], Rule('git', enabled_by_default=False), False),
 | 
			
		||||
        (const.DEFAULT_RULES, [], Rule('git', enabled_by_default=True), True),
 | 
			
		||||
        (const.DEFAULT_RULES, [], Rule('git', enabled_by_default=False), False),
 | 
			
		||||
        ([], [], Rule('git', enabled_by_default=False), False),
 | 
			
		||||
        ([], [], Rule('git', enabled_by_default=True), False),
 | 
			
		||||
        (conf.DEFAULT_RULES + ['git'], [], Rule('git', enabled_by_default=False), True),
 | 
			
		||||
        (const.DEFAULT_RULES + ['git'], [], Rule('git', enabled_by_default=False), True),
 | 
			
		||||
        (['git'], [], Rule('git', enabled_by_default=False), True),
 | 
			
		||||
        (conf.DEFAULT_RULES, ['git'], Rule('git', enabled_by_default=True), False),
 | 
			
		||||
        (conf.DEFAULT_RULES, ['git'], Rule('git', enabled_by_default=False), False),
 | 
			
		||||
        (const.DEFAULT_RULES, ['git'], Rule('git', enabled_by_default=True), False),
 | 
			
		||||
        (const.DEFAULT_RULES, ['git'], Rule('git', enabled_by_default=False), False),
 | 
			
		||||
        ([], ['git'], Rule('git', enabled_by_default=True), False),
 | 
			
		||||
        ([], ['git'], Rule('git', enabled_by_default=False), False)])
 | 
			
		||||
    def test_is_enabled(self, settings, rules, exclude_rules, rule, is_enabled):
 | 
			
		||||
@@ -95,11 +105,6 @@ class TestCommand(object):
 | 
			
		||||
        monkeypatch.setattr('thefuck.types.Command._wait_output',
 | 
			
		||||
                            staticmethod(lambda *_: True))
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def generic_shell(self, monkeypatch):
 | 
			
		||||
        monkeypatch.setattr('thefuck.shells.from_shell', lambda x: x)
 | 
			
		||||
        monkeypatch.setattr('thefuck.shells.to_shell', lambda x: x)
 | 
			
		||||
 | 
			
		||||
    def test_from_script_calls(self, Popen, settings):
 | 
			
		||||
        settings.env = {}
 | 
			
		||||
        assert Command.from_raw_script(
 | 
			
		||||
@@ -107,6 +112,7 @@ class TestCommand(object):
 | 
			
		||||
            'apt-get search vim', 'stdout', 'stderr')
 | 
			
		||||
        Popen.assert_called_once_with('apt-get search vim',
 | 
			
		||||
                                      shell=True,
 | 
			
		||||
                                      stdin=PIPE,
 | 
			
		||||
                                      stdout=PIPE,
 | 
			
		||||
                                      stderr=PIPE,
 | 
			
		||||
                                      env={})
 | 
			
		||||
@@ -122,4 +128,3 @@ class TestCommand(object):
 | 
			
		||||
        else:
 | 
			
		||||
            with pytest.raises(EmptyCommand):
 | 
			
		||||
                Command.from_raw_script(script)
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -4,39 +4,36 @@ import pytest
 | 
			
		||||
from itertools import islice
 | 
			
		||||
from thefuck import ui
 | 
			
		||||
from thefuck.types import CorrectedCommand
 | 
			
		||||
from thefuck import const
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def patch_getch(monkeypatch):
 | 
			
		||||
def patch_get_key(monkeypatch):
 | 
			
		||||
    def patch(vals):
 | 
			
		||||
        def getch():
 | 
			
		||||
            for val in vals:
 | 
			
		||||
                if val == KeyboardInterrupt:
 | 
			
		||||
                    raise val
 | 
			
		||||
                else:
 | 
			
		||||
                    yield val
 | 
			
		||||
 | 
			
		||||
        getch_gen = getch()
 | 
			
		||||
        monkeypatch.setattr('thefuck.ui.getch', lambda: next(getch_gen))
 | 
			
		||||
        vals = iter(vals)
 | 
			
		||||
        monkeypatch.setattr('thefuck.ui.get_key', lambda: next(vals))
 | 
			
		||||
 | 
			
		||||
    return patch
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def test_read_actions(patch_getch):
 | 
			
		||||
    patch_getch([  # Enter:
 | 
			
		||||
                   '\n',
 | 
			
		||||
                   # Enter:
 | 
			
		||||
                   '\r',
 | 
			
		||||
                   # Ignored:
 | 
			
		||||
                   'x', 'y',
 | 
			
		||||
                   # Up:
 | 
			
		||||
                   '\x1b', '[', 'A',
 | 
			
		||||
                   # Down:
 | 
			
		||||
                   '\x1b', '[', 'B',
 | 
			
		||||
                   # Ctrl+C:
 | 
			
		||||
                   KeyboardInterrupt], )
 | 
			
		||||
def test_read_actions(patch_get_key):
 | 
			
		||||
    patch_get_key([
 | 
			
		||||
        # Enter:
 | 
			
		||||
        '\n',
 | 
			
		||||
        # Enter:
 | 
			
		||||
        '\r',
 | 
			
		||||
        # Ignored:
 | 
			
		||||
        'x', 'y',
 | 
			
		||||
        # Up:
 | 
			
		||||
        const.KEY_UP,
 | 
			
		||||
        # Down:
 | 
			
		||||
        const.KEY_DOWN,
 | 
			
		||||
        # Ctrl+C:
 | 
			
		||||
        const.KEY_CTRL_C])
 | 
			
		||||
    assert list(islice(ui.read_actions(), 5)) \
 | 
			
		||||
           == [ui.SELECT, ui.SELECT, ui.PREVIOUS, ui.NEXT, ui.ABORT]
 | 
			
		||||
           == [const.ACTION_SELECT, const.ACTION_SELECT,
 | 
			
		||||
               const.ACTION_PREVIOUS, const.ACTION_NEXT,
 | 
			
		||||
               const.ACTION_ABORT]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def test_command_selector():
 | 
			
		||||
@@ -80,25 +77,25 @@ class TestSelectCommand(object):
 | 
			
		||||
               == commands_with_side_effect[0]
 | 
			
		||||
        assert capsys.readouterr() == ('', 'ls (+side effect)\n')
 | 
			
		||||
 | 
			
		||||
    def test_with_confirmation(self, capsys, patch_getch, commands):
 | 
			
		||||
        patch_getch(['\n'])
 | 
			
		||||
    def test_with_confirmation(self, capsys, patch_get_key, commands):
 | 
			
		||||
        patch_get_key(['\n'])
 | 
			
		||||
        assert ui.select_command(iter(commands)) == commands[0]
 | 
			
		||||
        assert capsys.readouterr() == ('', u'\x1b[1K\rls [enter/↑/↓/ctrl+c]\n')
 | 
			
		||||
 | 
			
		||||
    def test_with_confirmation_abort(self, capsys, patch_getch, commands):
 | 
			
		||||
        patch_getch([KeyboardInterrupt])
 | 
			
		||||
    def test_with_confirmation_abort(self, capsys, patch_get_key, commands):
 | 
			
		||||
        patch_get_key([const.KEY_CTRL_C])
 | 
			
		||||
        assert ui.select_command(iter(commands)) is None
 | 
			
		||||
        assert capsys.readouterr() == ('', u'\x1b[1K\rls [enter/↑/↓/ctrl+c]\nAborted\n')
 | 
			
		||||
 | 
			
		||||
    def test_with_confirmation_with_side_effct(self, capsys, patch_getch,
 | 
			
		||||
    def test_with_confirmation_with_side_effct(self, capsys, patch_get_key,
 | 
			
		||||
                                               commands_with_side_effect):
 | 
			
		||||
        patch_getch(['\n'])
 | 
			
		||||
        assert ui.select_command(iter(commands_with_side_effect))\
 | 
			
		||||
        patch_get_key(['\n'])
 | 
			
		||||
        assert ui.select_command(iter(commands_with_side_effect)) \
 | 
			
		||||
               == commands_with_side_effect[0]
 | 
			
		||||
        assert capsys.readouterr() == ('', u'\x1b[1K\rls (+side effect) [enter/↑/↓/ctrl+c]\n')
 | 
			
		||||
 | 
			
		||||
    def test_with_confirmation_select_second(self, capsys, patch_getch, commands):
 | 
			
		||||
        patch_getch(['\x1b', '[', 'B', '\n'])
 | 
			
		||||
    def test_with_confirmation_select_second(self, capsys, patch_get_key, commands):
 | 
			
		||||
        patch_get_key([const.KEY_DOWN, '\n'])
 | 
			
		||||
        assert ui.select_command(iter(commands)) == commands[1]
 | 
			
		||||
        assert capsys.readouterr() == (
 | 
			
		||||
            '', u'\x1b[1K\rls [enter/↑/↓/ctrl+c]\x1b[1K\rcd [enter/↑/↓/ctrl+c]\n')
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,8 @@ from mock import Mock
 | 
			
		||||
import six
 | 
			
		||||
from thefuck.utils import default_settings, \
 | 
			
		||||
    memoize, get_closest, get_all_executables, replace_argument, \
 | 
			
		||||
    get_all_matched_commands, is_app, for_app, cache, compatibility_call
 | 
			
		||||
    get_all_matched_commands, is_app, for_app, cache, compatibility_call, \
 | 
			
		||||
    get_valid_history_without_current
 | 
			
		||||
from tests.utils import Command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -50,7 +51,7 @@ class TestGetClosest(object):
 | 
			
		||||
 | 
			
		||||
@pytest.fixture
 | 
			
		||||
def get_aliases(mocker):
 | 
			
		||||
    mocker.patch('thefuck.shells.get_aliases',
 | 
			
		||||
    mocker.patch('thefuck.shells.shell.get_aliases',
 | 
			
		||||
                 return_value=['vim', 'apt-get', 'fsck', 'fuck'])
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -162,10 +163,10 @@ class TestCache(object):
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture
 | 
			
		||||
    def key(self):
 | 
			
		||||
        if six.PY3:
 | 
			
		||||
            return 'tests.test_utils.<function TestCache.fn.<locals>.fn '
 | 
			
		||||
        else:
 | 
			
		||||
        if six.PY2:
 | 
			
		||||
            return 'tests.test_utils.<function fn '
 | 
			
		||||
        else:
 | 
			
		||||
            return 'tests.test_utils.<function TestCache.fn.<locals>.fn '
 | 
			
		||||
 | 
			
		||||
    def test_with_blank_cache(self, shelve, fn, key):
 | 
			
		||||
        assert shelve == {}
 | 
			
		||||
@@ -229,3 +230,29 @@ class TestCompatibilityCall(object):
 | 
			
		||||
            return True
 | 
			
		||||
 | 
			
		||||
        assert compatibility_call(side_effect, Command(), Command())
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TestGetValidHistoryWithoutCurrent(object):
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def history(self, mocker):
 | 
			
		||||
        return mocker.patch('thefuck.shells.shell.get_history',
 | 
			
		||||
                            return_value=['le cat', 'fuck', 'ls cat',
 | 
			
		||||
                                          'diff x', 'nocommand x'])
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def alias(self, mocker):
 | 
			
		||||
        return mocker.patch('thefuck.utils.get_alias',
 | 
			
		||||
                            return_value='fuck')
 | 
			
		||||
 | 
			
		||||
    @pytest.fixture(autouse=True)
 | 
			
		||||
    def callables(self, mocker):
 | 
			
		||||
        return mocker.patch('thefuck.utils.get_all_executables',
 | 
			
		||||
                            return_value=['diff', 'ls'])
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.parametrize('script, result', [
 | 
			
		||||
        ('le cat', ['ls cat', 'diff x']),
 | 
			
		||||
        ('diff x', ['ls cat']),
 | 
			
		||||
        ('fuck', ['ls cat', 'diff x'])])
 | 
			
		||||
    def test_get_valid_history_without_current(self, script, result):
 | 
			
		||||
        command = Command(script=script)
 | 
			
		||||
        assert get_valid_history_without_current(command) == result
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
from thefuck import types
 | 
			
		||||
from thefuck.conf import DEFAULT_PRIORITY
 | 
			
		||||
from thefuck.const import DEFAULT_PRIORITY
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Command(types.Command):
 | 
			
		||||
 
 | 
			
		||||
@@ -3,40 +3,7 @@ import os
 | 
			
		||||
import sys
 | 
			
		||||
from pathlib import Path
 | 
			
		||||
from six import text_type
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ALL_ENABLED = object()
 | 
			
		||||
DEFAULT_RULES = [ALL_ENABLED]
 | 
			
		||||
DEFAULT_PRIORITY = 1000
 | 
			
		||||
 | 
			
		||||
DEFAULT_SETTINGS = {'rules': DEFAULT_RULES,
 | 
			
		||||
                    'exclude_rules': [],
 | 
			
		||||
                    'wait_command': 3,
 | 
			
		||||
                    'require_confirmation': True,
 | 
			
		||||
                    'no_colors': False,
 | 
			
		||||
                    'debug': False,
 | 
			
		||||
                    'priority': {},
 | 
			
		||||
                    'env': {'LC_ALL': 'C', 'LANG': 'C', 'GIT_TRACE': '1'}}
 | 
			
		||||
 | 
			
		||||
ENV_TO_ATTR = {'THEFUCK_RULES': 'rules',
 | 
			
		||||
               'THEFUCK_EXCLUDE_RULES': 'exclude_rules',
 | 
			
		||||
               'THEFUCK_WAIT_COMMAND': 'wait_command',
 | 
			
		||||
               'THEFUCK_REQUIRE_CONFIRMATION': 'require_confirmation',
 | 
			
		||||
               'THEFUCK_NO_COLORS': 'no_colors',
 | 
			
		||||
               'THEFUCK_PRIORITY': 'priority',
 | 
			
		||||
               'THEFUCK_DEBUG': 'debug'}
 | 
			
		||||
 | 
			
		||||
SETTINGS_HEADER = u"""# ~/.thefuck/settings.py: The Fuck settings file
 | 
			
		||||
#
 | 
			
		||||
# The rules are defined as in the example bellow:
 | 
			
		||||
#
 | 
			
		||||
# rules = ['cd_parent', 'git_push', 'python_command', 'sudo']
 | 
			
		||||
#
 | 
			
		||||
# The default values are as follows. Uncomment and change to fit your needs.
 | 
			
		||||
# See https://github.com/nvbn/thefuck#settings for more information.
 | 
			
		||||
#
 | 
			
		||||
 | 
			
		||||
"""
 | 
			
		||||
from . import const
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Settings(dict):
 | 
			
		||||
@@ -67,13 +34,25 @@ class Settings(dict):
 | 
			
		||||
        settings_path = self.user_dir.joinpath('settings.py')
 | 
			
		||||
        if not settings_path.is_file():
 | 
			
		||||
            with settings_path.open(mode='w') as settings_file:
 | 
			
		||||
                settings_file.write(SETTINGS_HEADER)
 | 
			
		||||
                for setting in DEFAULT_SETTINGS.items():
 | 
			
		||||
                settings_file.write(const.SETTINGS_HEADER)
 | 
			
		||||
                for setting in const.DEFAULT_SETTINGS.items():
 | 
			
		||||
                    settings_file.write(u'# {} = {}\n'.format(*setting))
 | 
			
		||||
 | 
			
		||||
    def _get_user_dir_path(self):
 | 
			
		||||
        # for backward compatibility, use `~/.thefuck` if it exists
 | 
			
		||||
        legacy_user_dir = Path(os.path.expanduser('~/.thefuck'))
 | 
			
		||||
 | 
			
		||||
        if legacy_user_dir.is_dir():
 | 
			
		||||
            return legacy_user_dir
 | 
			
		||||
        else:
 | 
			
		||||
            default_xdg_config_dir = os.path.expanduser("~/.config")
 | 
			
		||||
            xdg_config_dir = os.getenv("XDG_CONFIG_HOME", default_xdg_config_dir)
 | 
			
		||||
            return Path(os.path.join(xdg_config_dir, 'thefuck'))
 | 
			
		||||
 | 
			
		||||
    def _setup_user_dir(self):
 | 
			
		||||
        """Returns user config dir, create it when it doesn't exist."""
 | 
			
		||||
        user_dir = Path(os.path.expanduser('~/.thefuck'))
 | 
			
		||||
        user_dir = self._get_user_dir_path()
 | 
			
		||||
 | 
			
		||||
        rules_dir = user_dir.joinpath('rules')
 | 
			
		||||
        if not rules_dir.is_dir():
 | 
			
		||||
            rules_dir.mkdir(parents=True)
 | 
			
		||||
@@ -84,14 +63,14 @@ class Settings(dict):
 | 
			
		||||
        settings = load_source(
 | 
			
		||||
            'settings', text_type(self.user_dir.joinpath('settings.py')))
 | 
			
		||||
        return {key: getattr(settings, key)
 | 
			
		||||
                for key in DEFAULT_SETTINGS.keys()
 | 
			
		||||
                for key in const.DEFAULT_SETTINGS.keys()
 | 
			
		||||
                if hasattr(settings, key)}
 | 
			
		||||
 | 
			
		||||
    def _rules_from_env(self, val):
 | 
			
		||||
        """Transforms rules list from env-string to python."""
 | 
			
		||||
        val = val.split(':')
 | 
			
		||||
        if 'DEFAULT_RULES' in val:
 | 
			
		||||
            val = DEFAULT_RULES + [rule for rule in val if rule != 'DEFAULT_RULES']
 | 
			
		||||
            val = const.DEFAULT_RULES + [rule for rule in val if rule != 'DEFAULT_RULES']
 | 
			
		||||
        return val
 | 
			
		||||
 | 
			
		||||
    def _priority_from_env(self, val):
 | 
			
		||||
@@ -112,16 +91,19 @@ class Settings(dict):
 | 
			
		||||
            return dict(self._priority_from_env(val))
 | 
			
		||||
        elif attr == 'wait_command':
 | 
			
		||||
            return int(val)
 | 
			
		||||
        elif attr in ('require_confirmation', 'no_colors', 'debug'):
 | 
			
		||||
        elif attr in ('require_confirmation', 'no_colors', 'debug',
 | 
			
		||||
                      'alter_history'):
 | 
			
		||||
            return val.lower() == 'true'
 | 
			
		||||
        elif attr == 'history_limit':
 | 
			
		||||
            return int(val)
 | 
			
		||||
        else:
 | 
			
		||||
            return val
 | 
			
		||||
 | 
			
		||||
    def _settings_from_env(self):
 | 
			
		||||
        """Loads settings from env."""
 | 
			
		||||
        return {attr: self._val_from_env(env, attr)
 | 
			
		||||
                for env, attr in ENV_TO_ATTR.items()
 | 
			
		||||
                for env, attr in const.ENV_TO_ATTR.items()
 | 
			
		||||
                if env in os.environ}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
settings = Settings(DEFAULT_SETTINGS)
 | 
			
		||||
settings = Settings(const.DEFAULT_SETTINGS)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										56
									
								
								thefuck/const.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								thefuck/const.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,56 @@
 | 
			
		||||
# -*- encoding: utf-8 -*-
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class _GenConst(object):
 | 
			
		||||
    def __init__(self, name):
 | 
			
		||||
        self._name = name
 | 
			
		||||
 | 
			
		||||
    def __repr__(self):
 | 
			
		||||
        return u'<const: {}>'.format(self._name)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
KEY_UP = _GenConst('↑')
 | 
			
		||||
KEY_DOWN = _GenConst('↓')
 | 
			
		||||
KEY_CTRL_C = _GenConst('Ctrl+C')
 | 
			
		||||
 | 
			
		||||
ACTION_SELECT = _GenConst('select')
 | 
			
		||||
ACTION_ABORT = _GenConst('abort')
 | 
			
		||||
ACTION_PREVIOUS = _GenConst('previous')
 | 
			
		||||
ACTION_NEXT = _GenConst('next')
 | 
			
		||||
 | 
			
		||||
ALL_ENABLED = _GenConst('All rules enabled')
 | 
			
		||||
DEFAULT_RULES = [ALL_ENABLED]
 | 
			
		||||
DEFAULT_PRIORITY = 1000
 | 
			
		||||
 | 
			
		||||
DEFAULT_SETTINGS = {'rules': DEFAULT_RULES,
 | 
			
		||||
                    'exclude_rules': [],
 | 
			
		||||
                    'wait_command': 3,
 | 
			
		||||
                    'require_confirmation': True,
 | 
			
		||||
                    'no_colors': False,
 | 
			
		||||
                    'debug': False,
 | 
			
		||||
                    'priority': {},
 | 
			
		||||
                    'history_limit': None,
 | 
			
		||||
                    'alter_history': True,
 | 
			
		||||
                    'env': {'LC_ALL': 'C', 'LANG': 'C', 'GIT_TRACE': '1'}}
 | 
			
		||||
 | 
			
		||||
ENV_TO_ATTR = {'THEFUCK_RULES': 'rules',
 | 
			
		||||
               'THEFUCK_EXCLUDE_RULES': 'exclude_rules',
 | 
			
		||||
               'THEFUCK_WAIT_COMMAND': 'wait_command',
 | 
			
		||||
               'THEFUCK_REQUIRE_CONFIRMATION': 'require_confirmation',
 | 
			
		||||
               'THEFUCK_NO_COLORS': 'no_colors',
 | 
			
		||||
               'THEFUCK_DEBUG': 'debug',
 | 
			
		||||
               'THEFUCK_PRIORITY': 'priority',
 | 
			
		||||
               'THEFUCK_HISTORY_LIMIT': 'history_limit',
 | 
			
		||||
               'THEFUCK_ALTER_HISTORY': 'alter_history'}
 | 
			
		||||
 | 
			
		||||
SETTINGS_HEADER = u"""# The Fuck settings file
 | 
			
		||||
#
 | 
			
		||||
# The rules are defined as in the example bellow:
 | 
			
		||||
#
 | 
			
		||||
# rules = ['cd_parent', 'git_push', 'python_command', 'sudo']
 | 
			
		||||
#
 | 
			
		||||
# The default values are as follows. Uncomment and change to fit your needs.
 | 
			
		||||
# See https://github.com/nvbn/thefuck#settings for more information.
 | 
			
		||||
#
 | 
			
		||||
 | 
			
		||||
"""
 | 
			
		||||
@@ -55,7 +55,7 @@ def organize_commands(corrected_commands):
 | 
			
		||||
        key=lambda corrected_command: corrected_command.priority)
 | 
			
		||||
 | 
			
		||||
    logs.debug('Corrected commands: '.format(
 | 
			
		||||
        ', '.join(str(cmd) for cmd in [first_command] + sorted_commands)))
 | 
			
		||||
        ', '.join(u'{}'.format(cmd) for cmd in [first_command] + sorted_commands)))
 | 
			
		||||
 | 
			
		||||
    for command in sorted_commands:
 | 
			
		||||
        yield command
 | 
			
		||||
 
 | 
			
		||||
@@ -28,29 +28,29 @@ def exception(title, exc_info):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def rule_failed(rule, exc_info):
 | 
			
		||||
    exception('Rule {}'.format(rule.name), exc_info)
 | 
			
		||||
    exception(u'Rule {}'.format(rule.name), exc_info)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def failed(msg):
 | 
			
		||||
    sys.stderr.write('{red}{msg}{reset}\n'.format(
 | 
			
		||||
    sys.stderr.write(u'{red}{msg}{reset}\n'.format(
 | 
			
		||||
        msg=msg,
 | 
			
		||||
        red=color(colorama.Fore.RED),
 | 
			
		||||
        reset=color(colorama.Style.RESET_ALL)))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def show_corrected_command(corrected_command):
 | 
			
		||||
    sys.stderr.write('{bold}{script}{reset}{side_effect}\n'.format(
 | 
			
		||||
    sys.stderr.write(u'{bold}{script}{reset}{side_effect}\n'.format(
 | 
			
		||||
        script=corrected_command.script,
 | 
			
		||||
        side_effect=' (+side effect)' if corrected_command.side_effect else '',
 | 
			
		||||
        side_effect=u' (+side effect)' if corrected_command.side_effect else u'',
 | 
			
		||||
        bold=color(colorama.Style.BRIGHT),
 | 
			
		||||
        reset=color(colorama.Style.RESET_ALL)))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def confirm_text(corrected_command):
 | 
			
		||||
    sys.stderr.write(
 | 
			
		||||
        ('{clear}{bold}{script}{reset}{side_effect} '
 | 
			
		||||
         '[{green}enter{reset}/{blue}↑{reset}/{blue}↓{reset}'
 | 
			
		||||
         '/{red}ctrl+c{reset}]').format(
 | 
			
		||||
        (u'{clear}{bold}{script}{reset}{side_effect} '
 | 
			
		||||
         u'[{green}enter{reset}/{blue}↑{reset}/{blue}↓{reset}'
 | 
			
		||||
         u'/{red}ctrl+c{reset}]').format(
 | 
			
		||||
            script=corrected_command.script,
 | 
			
		||||
            side_effect=' (+side effect)' if corrected_command.side_effect else '',
 | 
			
		||||
            clear='\033[1K\r',
 | 
			
		||||
 
 | 
			
		||||
@@ -1,19 +1,23 @@
 | 
			
		||||
# Initialize output before importing any module, that can use colorama.
 | 
			
		||||
from .system import init_output
 | 
			
		||||
 | 
			
		||||
init_output()
 | 
			
		||||
 | 
			
		||||
from argparse import ArgumentParser
 | 
			
		||||
from warnings import warn
 | 
			
		||||
from pprint import pformat
 | 
			
		||||
import sys
 | 
			
		||||
import colorama
 | 
			
		||||
from . import logs, types, shells
 | 
			
		||||
from . import logs, types
 | 
			
		||||
from .shells import shell
 | 
			
		||||
from .conf import settings
 | 
			
		||||
from .corrector import get_corrected_commands
 | 
			
		||||
from .exceptions import EmptyCommand
 | 
			
		||||
from .utils import get_installation_info
 | 
			
		||||
from .utils import get_installation_info, get_alias
 | 
			
		||||
from .ui import select_command
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def fix_command():
 | 
			
		||||
    """Fixes previous command. Used when `thefuck` called without arguments."""
 | 
			
		||||
    colorama.init()
 | 
			
		||||
    settings.init()
 | 
			
		||||
    with logs.debug_time('Total'):
 | 
			
		||||
        logs.debug(u'Run with settings: {}'.format(pformat(settings)))
 | 
			
		||||
@@ -39,10 +43,10 @@ def print_alias(entry_point=True):
 | 
			
		||||
    else:
 | 
			
		||||
        position = 2
 | 
			
		||||
 | 
			
		||||
    alias = shells.thefuck_alias()
 | 
			
		||||
    alias = get_alias()
 | 
			
		||||
    if len(sys.argv) > position:
 | 
			
		||||
        alias = sys.argv[position]
 | 
			
		||||
    print(shells.app_alias(alias))
 | 
			
		||||
    print(shell.app_alias(alias))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def how_to_configure_alias():
 | 
			
		||||
@@ -51,19 +55,18 @@ def how_to_configure_alias():
 | 
			
		||||
    It'll be only visible when user type fuck and when alias isn't configured.
 | 
			
		||||
 | 
			
		||||
    """
 | 
			
		||||
    colorama.init()
 | 
			
		||||
    settings.init()
 | 
			
		||||
    logs.how_to_configure_alias(shells.how_to_configure())
 | 
			
		||||
    logs.how_to_configure_alias(shell.how_to_configure())
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def main():
 | 
			
		||||
    parser = ArgumentParser(prog='thefuck')
 | 
			
		||||
    version = get_installation_info().version
 | 
			
		||||
    parser.add_argument(
 | 
			
		||||
        '-v', '--version',
 | 
			
		||||
        action='version',
 | 
			
		||||
        version='The Fuck {} using Python {}'.format(
 | 
			
		||||
            version, sys.version.split()[0]))
 | 
			
		||||
            '-v', '--version',
 | 
			
		||||
            action='version',
 | 
			
		||||
            version='The Fuck {} using Python {}'.format(
 | 
			
		||||
                    version, sys.version.split()[0]))
 | 
			
		||||
    parser.add_argument('-a', '--alias',
 | 
			
		||||
                        action='store_true',
 | 
			
		||||
                        help='[custom-alias-name] prints alias for current shell')
 | 
			
		||||
 
 | 
			
		||||
@@ -1,8 +1,10 @@
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck.specific.apt import apt_available
 | 
			
		||||
from thefuck.utils import memoize
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
 | 
			
		||||
try:
 | 
			
		||||
    import CommandNotFound
 | 
			
		||||
    enabled_by_default = apt_available
 | 
			
		||||
except ImportError:
 | 
			
		||||
    enabled_by_default = False
 | 
			
		||||
 | 
			
		||||
@@ -26,5 +28,5 @@ def match(command):
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    name = get_package(command.script)
 | 
			
		||||
    formatme = shells.and_('sudo apt-get install {}', '{}')
 | 
			
		||||
    formatme = shell.and_('sudo apt-get install {}', '{}')
 | 
			
		||||
    return formatme.format(name, command.script)
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,9 @@
 | 
			
		||||
import re
 | 
			
		||||
from thefuck.specific.apt import apt_available
 | 
			
		||||
from thefuck.utils import for_app
 | 
			
		||||
 | 
			
		||||
enabled_by_default = apt_available
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@for_app('apt-get')
 | 
			
		||||
def match(command):
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										56
									
								
								thefuck/rules/apt_invalid_operation.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								thefuck/rules/apt_invalid_operation.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,56 @@
 | 
			
		||||
import subprocess
 | 
			
		||||
from thefuck.specific.apt import apt_available
 | 
			
		||||
from thefuck.specific.sudo import sudo_support
 | 
			
		||||
from thefuck.utils import for_app, eager, replace_command
 | 
			
		||||
 | 
			
		||||
enabled_by_default = apt_available
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@for_app('apt', 'apt-get', 'apt-cache')
 | 
			
		||||
@sudo_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    return 'E: Invalid operation' in command.stderr
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@eager
 | 
			
		||||
def _parse_apt_operations(help_text_lines):
 | 
			
		||||
    is_commands_list = False
 | 
			
		||||
    for line in help_text_lines:
 | 
			
		||||
        line = line.decode().strip()
 | 
			
		||||
        if is_commands_list and line:
 | 
			
		||||
            yield line.split()[0]
 | 
			
		||||
        elif line.startswith('Basic commands:'):
 | 
			
		||||
            is_commands_list = True
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@eager
 | 
			
		||||
def _parse_apt_get_and_cache_operations(help_text_lines):
 | 
			
		||||
    is_commands_list = False
 | 
			
		||||
    for line in help_text_lines:
 | 
			
		||||
        line = line.decode().strip()
 | 
			
		||||
        if is_commands_list:
 | 
			
		||||
            if not line:
 | 
			
		||||
                return
 | 
			
		||||
 | 
			
		||||
            yield line.split()[0]
 | 
			
		||||
        elif line.startswith('Commands:'):
 | 
			
		||||
            is_commands_list = True
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _get_operations(app):
 | 
			
		||||
    proc = subprocess.Popen([app, '--help'],
 | 
			
		||||
                            stdout=subprocess.PIPE,
 | 
			
		||||
                            stderr=subprocess.PIPE)
 | 
			
		||||
    lines = proc.stdout.readlines()
 | 
			
		||||
 | 
			
		||||
    if app == 'apt':
 | 
			
		||||
        return _parse_apt_operations(lines)
 | 
			
		||||
    else:
 | 
			
		||||
        return _parse_apt_get_and_cache_operations(lines)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@sudo_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    invalid_operation = command.stderr.split()[-1]
 | 
			
		||||
    operations = _get_operations(command.script_parts[0])
 | 
			
		||||
    return replace_command(command, invalid_operation, operations)
 | 
			
		||||
@@ -1,10 +1,9 @@
 | 
			
		||||
import os
 | 
			
		||||
import re
 | 
			
		||||
from thefuck.utils import get_closest, replace_argument, which
 | 
			
		||||
from thefuck.specific.brew import get_brew_path_prefix
 | 
			
		||||
from thefuck.utils import get_closest, replace_argument
 | 
			
		||||
from thefuck.specific.brew import get_brew_path_prefix, brew_available
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
enabled_by_default = bool(which('brew'))
 | 
			
		||||
enabled_by_default = brew_available
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _get_formulas():
 | 
			
		||||
 
 | 
			
		||||
@@ -1,12 +1,14 @@
 | 
			
		||||
import os
 | 
			
		||||
import re
 | 
			
		||||
from thefuck.utils import get_closest, replace_command
 | 
			
		||||
from thefuck.specific.brew import get_brew_path_prefix
 | 
			
		||||
from thefuck.specific.brew import get_brew_path_prefix, brew_available
 | 
			
		||||
 | 
			
		||||
BREW_CMD_PATH = '/Library/Homebrew/cmd'
 | 
			
		||||
TAP_PATH = '/Library/Taps'
 | 
			
		||||
TAP_CMD_PATH = '/%s/%s/cmd'
 | 
			
		||||
 | 
			
		||||
enabled_by_default = brew_available
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _get_brew_commands(brew_path_prefix):
 | 
			
		||||
    """To get brew default commands on local environment"""
 | 
			
		||||
@@ -53,7 +55,7 @@ def _brew_commands():
 | 
			
		||||
    if brew_path_prefix:
 | 
			
		||||
        try:
 | 
			
		||||
            return _get_brew_commands(brew_path_prefix) \
 | 
			
		||||
                 + _get_brew_tap_specific_commands(brew_path_prefix)
 | 
			
		||||
                   + _get_brew_tap_specific_commands(brew_path_prefix)
 | 
			
		||||
        except OSError:
 | 
			
		||||
            pass
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -4,6 +4,9 @@
 | 
			
		||||
# > brew upgrade
 | 
			
		||||
# Warning: brew upgrade with no arguments will change behaviour soon!
 | 
			
		||||
# It currently upgrades all formula but this will soon change to require '--all'.
 | 
			
		||||
from thefuck.specific.brew import brew_available
 | 
			
		||||
 | 
			
		||||
enabled_by_default = brew_available
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def match(command):
 | 
			
		||||
 
 | 
			
		||||
@@ -2,14 +2,14 @@ import re
 | 
			
		||||
from thefuck.utils import replace_argument, for_app
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@for_app('cargo')
 | 
			
		||||
@for_app('cargo', at_least=1)
 | 
			
		||||
def match(command):
 | 
			
		||||
    return ('No such subcommand' in command.stderr
 | 
			
		||||
            and 'Did you mean' in command.stderr)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    broken = command.script.split()[1]
 | 
			
		||||
    broken = command.script_parts[1]
 | 
			
		||||
    fix = re.findall(r'Did you mean `([^`]*)`', command.stderr)[0]
 | 
			
		||||
 | 
			
		||||
    return replace_argument(command.script, broken, fix)
 | 
			
		||||
 
 | 
			
		||||
@@ -1,11 +1,11 @@
 | 
			
		||||
"""Attempts to spellcheck and correct failed cd commands"""
 | 
			
		||||
 | 
			
		||||
import os
 | 
			
		||||
import six
 | 
			
		||||
from difflib import get_close_matches
 | 
			
		||||
import re
 | 
			
		||||
from thefuck.specific.sudo import sudo_support
 | 
			
		||||
from thefuck.rules import cd_mkdir
 | 
			
		||||
from thefuck.utils import for_app
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
 | 
			
		||||
__author__ = "mmussomele"
 | 
			
		||||
 | 
			
		||||
@@ -34,24 +34,25 @@ def get_new_command(command):
 | 
			
		||||
    defaults to the rules of cd_mkdir.
 | 
			
		||||
    Change sensitivity by changing MAX_ALLOWED_DIFF. Default value is 0.6
 | 
			
		||||
    """
 | 
			
		||||
    dest = command.script.split()[1].split(os.sep)
 | 
			
		||||
    dest = command.script_parts[1].split(os.sep)
 | 
			
		||||
    if dest[-1] == '':
 | 
			
		||||
        dest = dest[:-1]
 | 
			
		||||
    cwd = os.getcwd()
 | 
			
		||||
    if six.PY2:
 | 
			
		||||
        cwd = os.getcwdu()
 | 
			
		||||
    else:
 | 
			
		||||
        cwd = os.getcwd()
 | 
			
		||||
    for directory in dest:
 | 
			
		||||
        if directory == ".":
 | 
			
		||||
            continue
 | 
			
		||||
        elif directory == "..":
 | 
			
		||||
            cwd = os.path.split(cwd)[0]
 | 
			
		||||
            continue
 | 
			
		||||
        best_matches = get_close_matches(
 | 
			
		||||
            directory, _get_sub_dirs(cwd), cutoff=MAX_ALLOWED_DIFF)
 | 
			
		||||
        best_matches = get_close_matches(directory, _get_sub_dirs(cwd), cutoff=MAX_ALLOWED_DIFF)
 | 
			
		||||
        if best_matches:
 | 
			
		||||
            cwd = os.path.join(cwd, best_matches[0])
 | 
			
		||||
        else:
 | 
			
		||||
            repl = shells.and_('mkdir -p \\1', 'cd \\1')
 | 
			
		||||
            return re.sub(r'^cd (.*)', repl, command.script)
 | 
			
		||||
    return 'cd "{0}"'.format(cwd)
 | 
			
		||||
            return cd_mkdir.get_new_command(command)
 | 
			
		||||
    return u'cd "{0}"'.format(cwd)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
enabled_by_default = True
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										18
									
								
								thefuck/rules/cd_mkdir.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								thefuck/rules/cd_mkdir.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
			
		||||
import re
 | 
			
		||||
from thefuck.utils import for_app
 | 
			
		||||
from thefuck.specific.sudo import sudo_support
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@sudo_support
 | 
			
		||||
@for_app('cd')
 | 
			
		||||
def match(command):
 | 
			
		||||
    return (('no such file or directory' in command.stderr.lower()
 | 
			
		||||
             or 'cd: can\'t cd to' in command.stderr.lower()
 | 
			
		||||
             or 'the system cannot find the path specified.' in command.stderr.lower()))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@sudo_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    repl = shell.and_('mkdir -p \\1', 'cd \\1')
 | 
			
		||||
    return re.sub(r'^cd (.*)', repl, command.script)
 | 
			
		||||
@@ -1,7 +1,7 @@
 | 
			
		||||
from thefuck.utils import for_app
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@for_app(['g++', 'clang++'])
 | 
			
		||||
@for_app('g++', 'clang++')
 | 
			
		||||
def match(command):
 | 
			
		||||
    return ('This file requires compiler and library support for the '
 | 
			
		||||
            'ISO C++ 2011 standard.' in command.stderr or
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,12 @@
 | 
			
		||||
import tarfile
 | 
			
		||||
import os
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck.utils import for_app
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
tar_extensions = ('.tar', '.tar.Z', '.tar.bz2', '.tar.gz', '.tar.lz',
 | 
			
		||||
                  '.tar.lzma', '.tar.xz', '.taz', '.tb2', '.tbz', '.tbz2',
 | 
			
		||||
                  '.tgz', '.tlz', '.txz', '.tz')
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _is_tar_extract(cmd):
 | 
			
		||||
@@ -14,11 +19,7 @@ def _is_tar_extract(cmd):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _tar_file(cmd):
 | 
			
		||||
    tar_extensions = ('.tar', '.tar.Z', '.tar.bz2', '.tar.gz', '.tar.lz',
 | 
			
		||||
                      '.tar.lzma', '.tar.xz', '.taz', '.tb2', '.tbz', '.tbz2',
 | 
			
		||||
                      '.tgz', '.tlz', '.txz', '.tz')
 | 
			
		||||
 | 
			
		||||
    for c in cmd.split():
 | 
			
		||||
    for c in cmd:
 | 
			
		||||
        for ext in tar_extensions:
 | 
			
		||||
            if c.endswith(ext):
 | 
			
		||||
                return (c, c[0:len(c) - len(ext)])
 | 
			
		||||
@@ -28,16 +29,17 @@ def _tar_file(cmd):
 | 
			
		||||
def match(command):
 | 
			
		||||
    return ('-C' not in command.script
 | 
			
		||||
            and _is_tar_extract(command.script)
 | 
			
		||||
            and _tar_file(command.script) is not None)
 | 
			
		||||
            and _tar_file(command.script_parts) is not None)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return shells.and_('mkdir -p {dir}', '{cmd} -C {dir}') \
 | 
			
		||||
        .format(dir=_tar_file(command.script)[1], cmd=command.script)
 | 
			
		||||
    dir = shell.quote(_tar_file(command.script_parts)[1])
 | 
			
		||||
    return shell.and_('mkdir -p {dir}', '{cmd} -C {dir}') \
 | 
			
		||||
        .format(dir=dir, cmd=command.script)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def side_effect(old_cmd, command):
 | 
			
		||||
    with tarfile.TarFile(_tar_file(old_cmd.script)[0]) as archive:
 | 
			
		||||
    with tarfile.TarFile(_tar_file(old_cmd.script_parts)[0]) as archive:
 | 
			
		||||
        for file in archive.getnames():
 | 
			
		||||
            try:
 | 
			
		||||
                os.remove(file)
 | 
			
		||||
 
 | 
			
		||||
@@ -1,11 +1,15 @@
 | 
			
		||||
import os
 | 
			
		||||
import zipfile
 | 
			
		||||
from thefuck.utils import for_app
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _is_bad_zip(file):
 | 
			
		||||
    with zipfile.ZipFile(file, 'r') as archive:
 | 
			
		||||
        return len(archive.namelist()) > 1
 | 
			
		||||
    try:
 | 
			
		||||
        with zipfile.ZipFile(file, 'r') as archive:
 | 
			
		||||
            return len(archive.namelist()) > 1
 | 
			
		||||
    except:
 | 
			
		||||
        return False
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _zip_file(command):
 | 
			
		||||
@@ -13,22 +17,29 @@ def _zip_file(command):
 | 
			
		||||
    # unzip [-flags] file[.zip] [file(s) ...] [-x file(s) ...]
 | 
			
		||||
    #                ^          ^ files to unzip from the archive
 | 
			
		||||
    #                archive to unzip
 | 
			
		||||
    for c in command.script.split()[1:]:
 | 
			
		||||
    for c in command.script_parts[1:]:
 | 
			
		||||
        if not c.startswith('-'):
 | 
			
		||||
            if c.endswith('.zip'):
 | 
			
		||||
                return c
 | 
			
		||||
            else:
 | 
			
		||||
                return '{}.zip'.format(c)
 | 
			
		||||
                return u'{}.zip'.format(c)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@for_app('unzip')
 | 
			
		||||
def match(command):
 | 
			
		||||
    return ('-d' not in command.script
 | 
			
		||||
            and _is_bad_zip(_zip_file(command)))
 | 
			
		||||
    if '-d' in command.script:
 | 
			
		||||
        return False
 | 
			
		||||
 | 
			
		||||
    zip_file = _zip_file(command)
 | 
			
		||||
    if zip_file:
 | 
			
		||||
        return _is_bad_zip(zip_file)
 | 
			
		||||
    else:
 | 
			
		||||
        return False
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return '{} -d {}'.format(command.script, _zip_file(command)[:-4])
 | 
			
		||||
    return u'{} -d {}'.format(
 | 
			
		||||
            command.script, shell.quote(_zip_file(command)[:-4]))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def side_effect(old_cmd, command):
 | 
			
		||||
 
 | 
			
		||||
@@ -1,11 +1,13 @@
 | 
			
		||||
def match(command):
 | 
			
		||||
    split_command = command.script.split()
 | 
			
		||||
    split_command = command.script_parts
 | 
			
		||||
 | 
			
		||||
    return len(split_command) >= 2 and split_command[0] == split_command[1]
 | 
			
		||||
    return (split_command
 | 
			
		||||
            and len(split_command) >= 2
 | 
			
		||||
            and split_command[0] == split_command[1])
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return command.script[command.script.find(' ')+1:]
 | 
			
		||||
    return ' '.join(command.script_parts[1:])
 | 
			
		||||
 | 
			
		||||
# it should be rare enough to actually have to type twice the same word, so
 | 
			
		||||
# this rule can have a higher priority to come before things like "cd cd foo"
 | 
			
		||||
 
 | 
			
		||||
@@ -2,43 +2,43 @@ import re
 | 
			
		||||
import os
 | 
			
		||||
from thefuck.utils import memoize, default_settings
 | 
			
		||||
from thefuck.conf import settings
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# order is important: only the first match is considered
 | 
			
		||||
patterns = (
 | 
			
		||||
        # js, node:
 | 
			
		||||
        '^    at {file}:{line}:{col}',
 | 
			
		||||
        # cargo:
 | 
			
		||||
        '^   {file}:{line}:{col}',
 | 
			
		||||
        # python, thefuck:
 | 
			
		||||
        '^  File "{file}", line {line}',
 | 
			
		||||
        # awk:
 | 
			
		||||
        '^awk: {file}:{line}:',
 | 
			
		||||
        # git
 | 
			
		||||
        '^fatal: bad config file line {line} in {file}',
 | 
			
		||||
        # llc:
 | 
			
		||||
        '^llc: {file}:{line}:{col}:',
 | 
			
		||||
        # lua:
 | 
			
		||||
        '^lua: {file}:{line}:',
 | 
			
		||||
        # fish:
 | 
			
		||||
        '^{file} \\(line {line}\\):',
 | 
			
		||||
        # bash, sh, ssh:
 | 
			
		||||
        '^{file}: line {line}: ',
 | 
			
		||||
        # cargo, clang, gcc, go, pep8, rustc:
 | 
			
		||||
        '^{file}:{line}:{col}',
 | 
			
		||||
        # ghc, make, ruby, zsh:
 | 
			
		||||
        '^{file}:{line}:',
 | 
			
		||||
        # perl:
 | 
			
		||||
        'at {file} line {line}',
 | 
			
		||||
    )
 | 
			
		||||
    # js, node:
 | 
			
		||||
    '^    at {file}:{line}:{col}',
 | 
			
		||||
    # cargo:
 | 
			
		||||
    '^   {file}:{line}:{col}',
 | 
			
		||||
    # python, thefuck:
 | 
			
		||||
    '^  File "{file}", line {line}',
 | 
			
		||||
    # awk:
 | 
			
		||||
    '^awk: {file}:{line}:',
 | 
			
		||||
    # git
 | 
			
		||||
    '^fatal: bad config file line {line} in {file}',
 | 
			
		||||
    # llc:
 | 
			
		||||
    '^llc: {file}:{line}:{col}:',
 | 
			
		||||
    # lua:
 | 
			
		||||
    '^lua: {file}:{line}:',
 | 
			
		||||
    # fish:
 | 
			
		||||
    '^{file} \\(line {line}\\):',
 | 
			
		||||
    # bash, sh, ssh:
 | 
			
		||||
    '^{file}: line {line}: ',
 | 
			
		||||
    # cargo, clang, gcc, go, pep8, rustc:
 | 
			
		||||
    '^{file}:{line}:{col}',
 | 
			
		||||
    # ghc, make, ruby, zsh:
 | 
			
		||||
    '^{file}:{line}:',
 | 
			
		||||
    # perl:
 | 
			
		||||
    'at {file} line {line}',
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# for the sake of readability do not use named groups above
 | 
			
		||||
def _make_pattern(pattern):
 | 
			
		||||
    pattern = pattern.replace('{file}', '(?P<file>[^:\n]+)') \
 | 
			
		||||
                     .replace('{line}', '(?P<line>[0-9]+)') \
 | 
			
		||||
                     .replace('{col}',  '(?P<col>[0-9]+)')
 | 
			
		||||
                     .replace('{col}', '(?P<col>[0-9]+)')
 | 
			
		||||
    return re.compile(pattern, re.MULTILINE)
 | 
			
		||||
patterns = [_make_pattern(p).search for p in patterns]
 | 
			
		||||
 | 
			
		||||
@@ -58,7 +58,7 @@ def match(command):
 | 
			
		||||
    return _search(command.stderr) or _search(command.stdout)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@default_settings({'fixlinecmd': '{editor} {file} +{line}',
 | 
			
		||||
@default_settings({'fixlinecmd': u'{editor} {file} +{line}',
 | 
			
		||||
                   'fixcolcmd': None})
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    m = _search(command.stderr) or _search(command.stdout)
 | 
			
		||||
@@ -75,4 +75,4 @@ def get_new_command(command):
 | 
			
		||||
                                                 file=m.group('file'),
 | 
			
		||||
                                                 line=m.group('line'))
 | 
			
		||||
 | 
			
		||||
    return shells.and_(editor_call, command.script)
 | 
			
		||||
    return shell.and_(editor_call, command.script)
 | 
			
		||||
 
 | 
			
		||||
@@ -1,19 +1,18 @@
 | 
			
		||||
import re
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    return ('did not match any file(s) known to git.' in command.stderr
 | 
			
		||||
            and "Did you forget to 'git add'?" in command.stderr)
 | 
			
		||||
    return 'did not match any file(s) known to git.' in command.stderr
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    missing_file = re.findall(
 | 
			
		||||
            r"error: pathspec '([^']*)' "
 | 
			
		||||
            r"did not match any file\(s\) known to git.", command.stderr)[0]
 | 
			
		||||
        r"error: pathspec '([^']*)' "
 | 
			
		||||
        r'did not match any file\(s\) known to git.', command.stderr)[0]
 | 
			
		||||
 | 
			
		||||
    formatme = shells.and_('git add -- {}', '{}')
 | 
			
		||||
    formatme = shell.and_('git add -- {}', '{}')
 | 
			
		||||
    return formatme.format(missing_file, command.script)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										23
									
								
								thefuck/rules/git_branch_exists.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								thefuck/rules/git_branch_exists.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,23 @@
 | 
			
		||||
import re
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
from thefuck.utils import eager
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    return ('branch' in command.script
 | 
			
		||||
            and "fatal: A branch named '" in command.stderr
 | 
			
		||||
            and " already exists." in command.stderr)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
@eager
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    branch_name = re.findall(
 | 
			
		||||
        r"fatal: A branch named '([^']*)' already exists.", command.stderr)[0]
 | 
			
		||||
    new_command_templates = [['git branch -d {0}', 'git branch {0}'],
 | 
			
		||||
                             ['git branch -D {0}', 'git branch {0}'],
 | 
			
		||||
                             ['git checkout {0}']]
 | 
			
		||||
    for new_command_template in new_command_templates:
 | 
			
		||||
        yield shell.and_(*new_command_template).format(branch_name)
 | 
			
		||||
@@ -1,13 +1,14 @@
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    # catches "git branch list" in place of "git branch"
 | 
			
		||||
    return command.script.split()[1:] == 'branch list'.split()
 | 
			
		||||
    return (command.script_parts
 | 
			
		||||
            and command.script_parts[1:] == 'branch list'.split())
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return shells.and_('git branch --delete list', 'git branch')
 | 
			
		||||
    return shell.and_('git branch --delete list', 'git branch')
 | 
			
		||||
 
 | 
			
		||||
@@ -1,8 +1,9 @@
 | 
			
		||||
import re
 | 
			
		||||
import subprocess
 | 
			
		||||
from thefuck import shells, utils
 | 
			
		||||
from thefuck import utils
 | 
			
		||||
from thefuck.utils import replace_argument
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
@@ -34,5 +35,5 @@ def get_new_command(command):
 | 
			
		||||
    if closest_branch:
 | 
			
		||||
        return replace_argument(command.script, missing_file, closest_branch)
 | 
			
		||||
    else:
 | 
			
		||||
        return shells.and_('git branch {}', '{}').format(
 | 
			
		||||
        return shell.and_('git branch {}', '{}').format(
 | 
			
		||||
            missing_file, command.script)
 | 
			
		||||
 
 | 
			
		||||
@@ -5,9 +5,8 @@ from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    splited_script = command.script.split()
 | 
			
		||||
    if len(splited_script) > 1:
 | 
			
		||||
        return (splited_script[1] == 'stash'
 | 
			
		||||
    if command.script_parts and len(command.script_parts) > 1:
 | 
			
		||||
        return (command.script_parts[1] == 'stash'
 | 
			
		||||
                and 'usage:' in command.stderr)
 | 
			
		||||
    else:
 | 
			
		||||
        return False
 | 
			
		||||
@@ -26,12 +25,12 @@ stash_commands = (
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    stash_cmd = command.script.split()[2]
 | 
			
		||||
    stash_cmd = command.script_parts[2]
 | 
			
		||||
    fixed = utils.get_closest(stash_cmd, stash_commands, fallback_to_first=False)
 | 
			
		||||
 | 
			
		||||
    if fixed is not None:
 | 
			
		||||
        return replace_argument(command.script, stash_cmd, fixed)
 | 
			
		||||
    else:
 | 
			
		||||
        cmd = command.script.split()
 | 
			
		||||
        cmd = command.script_parts[:]
 | 
			
		||||
        cmd.insert(2, 'save')
 | 
			
		||||
        return ' '.join(cmd)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										12
									
								
								thefuck/rules/git_help_aliased.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								thefuck/rules/git_help_aliased.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,12 @@
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    return 'help' in command.script and ' is aliased to ' in command.stdout
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    aliased = command.stdout.split('`', 2)[2].split("'", 1)[0].split(' ', 1)[0]
 | 
			
		||||
    return 'git help {}'.format(aliased)
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -14,4 +14,4 @@ def get_new_command(command):
 | 
			
		||||
    branch = line.split(' ')[-1]
 | 
			
		||||
    set_upstream = line.replace('<remote>', 'origin')\
 | 
			
		||||
                       .replace('<branch>', branch)
 | 
			
		||||
    return shells.and_(set_upstream, command.script)
 | 
			
		||||
    return shell.and_(set_upstream, command.script)
 | 
			
		||||
 
 | 
			
		||||
@@ -12,7 +12,7 @@ def match(command):
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return replace_argument(command.script, 'push', 'push --force')
 | 
			
		||||
    return replace_argument(command.script, 'push', 'push --force-with-lease')
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
enabled_by_default = False
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
from thefuck.utils import replace_argument
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
@@ -13,5 +13,5 @@ def match(command):
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return shells.and_(replace_argument(command.script, 'push', 'pull'),
 | 
			
		||||
                       command.script)
 | 
			
		||||
    return shell.and_(replace_argument(command.script, 'push', 'pull'),
 | 
			
		||||
                      command.script)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										14
									
								
								thefuck/rules/git_remote_seturl_add.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								thefuck/rules/git_remote_seturl_add.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,14 @@
 | 
			
		||||
from thefuck.utils import replace_argument
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    return ('set-url' in command.script and 'fatal: No such remote' in command.stderr)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return replace_argument(command.script, 'set-url', 'add')
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
enabled_by_default = True
 | 
			
		||||
							
								
								
									
										15
									
								
								thefuck/rules/git_rm_recursive.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								thefuck/rules/git_rm_recursive.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,15 @@
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    return (' rm ' in command.script
 | 
			
		||||
            and "fatal: not removing '" in command.stderr
 | 
			
		||||
            and "' recursively without -r" in command.stderr)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    index = command.script_parts.index('rm') + 1
 | 
			
		||||
    command.script_parts.insert(index, '-r')
 | 
			
		||||
    return u' '.join(command.script_parts)
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
from thefuck import shells
 | 
			
		||||
from thefuck.shells import shell
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -11,5 +11,5 @@ def match(command):
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    formatme = shells.and_('git stash', '{}')
 | 
			
		||||
    formatme = shell.and_('git stash', '{}')
 | 
			
		||||
    return formatme.format(command.script)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										14
									
								
								thefuck/rules/git_two_dashes.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								thefuck/rules/git_two_dashes.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,14 @@
 | 
			
		||||
from thefuck.utils import replace_argument
 | 
			
		||||
from thefuck.specific.git import git_support
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    return ('error: did you mean `' in command.stderr
 | 
			
		||||
            and '` (with two dashes ?)' in command.stderr)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@git_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    to = command.stderr.split('`')[1]
 | 
			
		||||
    return replace_argument(command.script, to[1:], to)
 | 
			
		||||
							
								
								
									
										23
									
								
								thefuck/rules/grep_arguments_order.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								thefuck/rules/grep_arguments_order.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,23 @@
 | 
			
		||||
import os
 | 
			
		||||
from thefuck.utils import for_app
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _get_actual_file(parts):
 | 
			
		||||
    for part in parts[1:]:
 | 
			
		||||
        if os.path.isfile(part) or os.path.isdir(part):
 | 
			
		||||
            return part
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@for_app('grep', 'egrep')
 | 
			
		||||
def match(command):
 | 
			
		||||
    return ': No such file or directory' in command.stderr \
 | 
			
		||||
        and _get_actual_file(command.script_parts)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    actual_file = _get_actual_file(command.script_parts)
 | 
			
		||||
    parts = command.script_parts[::]
 | 
			
		||||
    # Moves file to the end of the script:
 | 
			
		||||
    parts.remove(actual_file)
 | 
			
		||||
    parts.append(actual_file)
 | 
			
		||||
    return ' '.join(parts)
 | 
			
		||||
@@ -7,4 +7,4 @@ def match(command):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return 'grep -r {}'.format(command.script[5:])
 | 
			
		||||
    return u'grep -r {}'.format(command.script[5:])
 | 
			
		||||
 
 | 
			
		||||
@@ -4,7 +4,7 @@ from thefuck.specific.sudo import sudo_support
 | 
			
		||||
 | 
			
		||||
@sudo_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    return os.path.exists(command.script.split()[0]) \
 | 
			
		||||
    return command.script_parts and os.path.exists(command.script_parts[0]) \
 | 
			
		||||
        and 'command not found' in command.stderr
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,37 +1,15 @@
 | 
			
		||||
from difflib import get_close_matches
 | 
			
		||||
from thefuck.shells import get_history, thefuck_alias
 | 
			
		||||
from thefuck.utils import get_closest, memoize, get_all_executables
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _not_corrected(history, tf_alias):
 | 
			
		||||
    """Returns all lines from history except that comes before `fuck`."""
 | 
			
		||||
    previous = None
 | 
			
		||||
    for line in history:
 | 
			
		||||
        if previous is not None and line != tf_alias:
 | 
			
		||||
            yield previous
 | 
			
		||||
        previous = line
 | 
			
		||||
    if history:
 | 
			
		||||
        yield history[-1]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@memoize
 | 
			
		||||
def _history_of_exists_without_current(command):
 | 
			
		||||
    history = get_history()
 | 
			
		||||
    tf_alias = thefuck_alias()
 | 
			
		||||
    executables = get_all_executables()
 | 
			
		||||
    return [line for line in _not_corrected(history, tf_alias)
 | 
			
		||||
            if not line.startswith(tf_alias) and not line == command.script
 | 
			
		||||
            and line.split(' ')[0] in executables]
 | 
			
		||||
from thefuck.utils import get_closest, get_valid_history_without_current
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def match(command):
 | 
			
		||||
    return len(get_close_matches(command.script,
 | 
			
		||||
                                 _history_of_exists_without_current(command)))
 | 
			
		||||
                                 get_valid_history_without_current(command)))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return get_closest(command.script,
 | 
			
		||||
                       _history_of_exists_without_current(command))
 | 
			
		||||
                       get_valid_history_without_current(command))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
priority = 9999
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										23
									
								
								thefuck/rules/ln_no_hard_link.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								thefuck/rules/ln_no_hard_link.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,23 @@
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
"""Suggest creating symbolic link if hard link is not allowed.
 | 
			
		||||
 | 
			
		||||
Example:
 | 
			
		||||
> ln barDir barLink
 | 
			
		||||
ln: ‘barDir’: hard link not allowed for directory
 | 
			
		||||
 | 
			
		||||
--> ln -s barDir barLink
 | 
			
		||||
"""
 | 
			
		||||
 | 
			
		||||
import re
 | 
			
		||||
from thefuck.specific.sudo import sudo_support
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@sudo_support
 | 
			
		||||
def match(command):
 | 
			
		||||
    return (command.stderr.endswith("hard link not allowed for directory") and
 | 
			
		||||
            command.script.startswith("ln "))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@sudo_support
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    return re.sub(r'^ln ', 'ln -s ', command.script)
 | 
			
		||||
@@ -3,10 +3,10 @@ from thefuck.utils import for_app
 | 
			
		||||
 | 
			
		||||
@for_app('ls')
 | 
			
		||||
def match(command):
 | 
			
		||||
    return 'ls -' not in command.script
 | 
			
		||||
    return command.script_parts and 'ls -' not in command.script
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_new_command(command):
 | 
			
		||||
    command = command.script.split(' ')
 | 
			
		||||
    command = command.script_parts[:]
 | 
			
		||||
    command[0] = 'ls -lah'
 | 
			
		||||
    return ' '.join(command)
 | 
			
		||||
 
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user