Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ssbr/refex
A syntactically aware search-and-replace tool for Python.
https://github.com/ssbr/refex
Last synced: 2 months ago
JSON representation
A syntactically aware search-and-replace tool for Python.
- Host: GitHub
- URL: https://github.com/ssbr/refex
- Owner: ssbr
- License: apache-2.0
- Created: 2020-08-27T19:33:42.000Z (over 4 years ago)
- Default Branch: main
- Last Pushed: 2023-09-27T20:48:32.000Z (over 1 year ago)
- Last Synced: 2024-08-02T00:23:02.116Z (5 months ago)
- Language: Python
- Size: 505 KB
- Stars: 14
- Watchers: 3
- Forks: 1
- Open Issues: 17
-
Metadata Files:
- Readme: README.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
Awesome Lists containing this project
- awesome-codemods - **refex** - refactor expressions in Python (Libraries/Tools for refactoring / Debian)
README
# Refex - refactoring expressions
Refex is a syntactically aware search-and-replace tool for Python, which allows you to specify code searches and rewrites using templates, or a
more complex
[Clang-LibASTMatcher](https://clang.llvm.org/docs/LibASTMatchersTutorial.html#intermezzo-learn-ast-matcher-basics)-like
matcher interface.## Examples
**Automatic parenthesis insertion:** Refex will automatically insert parentheses
to preserve the intended code structure:```sh
$ echo "a = b.foo() * c" > test.py
$ refex --mode=py.expr '$x.foo()' --sub='$x.foo() + 1' -i test.py
...
$ cat test.py
a = (b.foo() + 1) * c
```A naive regular expression replacement would have resulted in `b.foo() + 1 * c`, which is not
equivalent, and is unrelated to the intended replacement.**Paired parentheses:** Refex is aware of the full syntax tree, and will always match parentheses correctly:
```sh
$ echo "print(foo(bar(b))" > test.py
$ refex --mode=py.expr 'foo($x)' --sub='foo($x + 1)' -i test.py
...
$ cat test.py
a = print(foo(bar(b) + 1))
```Here, a naive replacement using regular expressions could have resulted in
either `print(foo(bar(b)) + 1)` or `print(foo(bar(b) + 1))`, depending on
whether `$x` was matched greedily or non-greedily.**Combining replacements:** you can pass multiple search/replace pairs to
Refex which combine to do more complex rewrites. For example:```sh
# Rewrites "self.assertTrue(x == False)" to "self.assertFalse(x)", even though
# that was not explicitly called out.
refex --mode=py.expr -i --iterate \
--match='self.assertTrue($x == $y)' --sub='self.assertEqual($x, $y)' \
--match='self.assertEqual($x, False)' --sub='self.assertFalse($x)' \
-R dir/
```TODO: also describe `--mode=py`.
## Getting started
### Installation
Refex can be run via [pipx](https://pipxproject.github.io/pipx/) for one-off use
with control over the Python version:```sh
$ pipx run refex --help
```For longer-term use, or for use of Refex [as a library](https://refex.readthedocs.io/en/latest/guide/library.html),
it is also pip-installable:```sh
$ python3 -m venv my_env
$ source my_env/bin/activate
$ pip install refex
$ refex --help
```### Use
The template syntax is almost exactly what it looks like, so the examples at the
top of this page, in combination with the `--help` output, are intended to be
enough to get started.For more details on the template syntax, see [Python Patterns and Templates](https://refex.readthedocs.io/en/latest/guide/patterns_templates.html). For details on how to use refex in your own code as a library, see [Using Refex as a Library](https://refex.readthedocs.io/en/latest/guide/library.html).
## Current status
**Stable:**
The APIs documented at https://refex.readthedocs.io/ are expected to remain
mostly the same, except for trivial renames and moves.These command-line interfaces are expected to remain roughly the same, without
backwards-incompatible changes:* `--mode=py.expr`
* `--mode=fix`
* `--mode=re`**Unstable**
* All undocumented APIs (*especially* the API for creating a new matcher).
* `--mode=py.stmt` is missing many safety and convenience features.
* `--mode=py`, the matcher interface, will eventually need some fairly large
restructuring to make it O(n), although simple uses should be unaffected.(Also, all the stable parts are unstable too. This isn't a promise, just an
expectation/statement of intent.)## Contributing
See the
[contribution guide](https://refex.readthedocs.io/en/latest/meta/contributing.html)## See Also
* [asttokens](https://github.com/gristlabs/asttokens): the token-preserving
AST library that Refex is built on top of.
* [Pasta](https://github.com/google/pasta): a code rewriting tool using AST
mutation instead of string templates.
* [Semgrep](https://github.com/returntocorp/semgrep): cross-language AST
search using a similar approach.
* [lib2to3](https://docs.python.org/3/library/2to3.html#module-lib2to3): the
standard library's code rewriting tool based on the concrete syntax tree.## Disclaimer
You may have noticed Google copyright notices. This is not an officially
supported Google product.