# Rector - Upgrade your Legacy App to Modern Codebase [![Build Status](https://img.shields.io/travis/rectorphp/rector/master.svg?style=flat-square)](https://travis-ci.org/rectorphp/rector) [![Coverage Status](https://img.shields.io/coveralls/RectorPHP/Rector/master.svg?style=flat-square)](https://coveralls.io/github/rectorphp/rector?branch=master) Rector **upgrades your application** for you, with focus on open-source projects:

It can: - rename class - rename class methods - rename partial namespace - rename "_" pseudo-namespace to namespace - add arguments - replace arguments - remove arguments - add typehints based on new types of parent class or interface - and more...
## Install ```bash composer require --dev rector/rector @dev nikic/php-parser '4.0.x-dev' ``` Do you have old PHP or dependencies in conflict? Ok, [it is not problem](/docs/HowUseWithOldPhp.md). ## How To Reconstruct your Code? ### A. Prepared Sets Featured open-source projects have **prepared sets**. You'll find them in [`/src/config/level`](/src/config/level). E.g. Do you need to upgrade to Symfony 4.0? 1. Run rector on your `/src` directory ```bash vendor/bin/rector process src --level symfony40 ``` Which is just a shortcut for using complete path with `--config` option: ```bash vendor/bin/rector process src --config vendor/rector/rector/src/config/level/symfony/symfony40.yml ``` You can also use your **own config file**: ```bash vendor/bin/rector process src --config your-own-config.yml ``` 2. Do you want to see preview of changes first? Use `--dry-run` option: ``` vendor/bin/rector process src --level symfony33 --dry-run ``` ### B. Custom Sets 1. Create `rector.yml` with desired Rectors ```yml rectors: - Rector\Rector\Contrib\Nette\Application\InjectPropertyRector ``` 2. Try Rector on your `/src` directory ```bash vendor/bin/rector process src --dry-run ``` 3. Apply Changes if you like them ``` vendor/bin/rector process src ``` ### Simple setup with Dynamic Rectors You don't have to always write PHP code. Many projects change only classes or method names, so it would be too much work for a simple task. Instead you can use prepared **Dynamic Rectors** directly in `*.yml` config: You can: - **replace class name** ```yml # phpunit60.yml rectors: Rector\Rector\Dynamic\ClassReplacerRector: # old class: new class 'PHPUnit_Framework_TestCase': 'PHPUnit\Framework\TestCase' ``` - **replace part of namespace** ```yml # better-reflection20.yml rectors: Rector\Rector\Dynamic\NamespaceReplacerRector: # old namespace: new namespace 'BetterReflection': 'Roave\BetterReflection' ``` - **change method name** ```yml rectors: Rector\Rector\Dynamic\MethodNameReplacerRector: # class 'Nette\Utils\Html': # old method: new method 'add': 'addHtml' # or in case of static methods calls # class 'Nette\Bridges\FormsLatte\FormMacros': # old method: [new class, new method] 'renderFormBegin': ['Nette\Bridges\FormsLatte\Runtime', 'renderFormBegin'] ``` - **change property name** ```yml rectors: Rector\Rector\Dynamic\PropertyNameReplacerRector: # class: 'PhpParser\Node\Param': # old property: new property 'name': 'var' ``` - **change class constant name** ```yml rectors: Rector\Rector\Dynamic\ClassConstantReplacerRector: # class 'Symfony\Component\Form\FormEvents': # old constant: new constant 'PRE_BIND': 'PRE_SUBMIT' 'BIND': 'SUBMIT' 'POST_BIND': 'POST_SUBMIT' ``` - **change parameters typehint according to parent type** ```yml rectors: Rector\Rector\Dynamic\ParentTypehintedArgumentRector: # class 'PhpParser\Parser': # method 'parse': # parameter: typehint 'code': 'string' ``` - **change argument value** ```yml rectors: Rector\Rector\Dynamic\ArgumentReplacerRector: # class 'Symfony\Component\DependencyInjection\ContainerBuilder': # method 'compile': # argument position 0: # added default value '~': false # or remove completely '~': ~ # or replace by new value 'Symfony\Component\DependencyInjection\ContainerBuilder\ContainerBuilder::SCOPE_PROTOTYPE': false ``` - or **replace underscore naming `_` with namespaces `\`** ```yml rectors: Rector\Rector\Dynamic\PseudoNamespaceToNamespaceRector: # old namespace prefix - 'PHPUnit_' # exclude classes - '!PHPUnit_Framework_MockObject_MockObject' ``` - or **change property to method** ```yml rectors: Rector\Rector\Dynamic\PropertyToMethodRector: # type 'Symfony\Component\Translation\Translator': # property to replace 'locale': # (prepared key): get method name 'get': 'getLocale' # (prepared key): set method name 'set': 'setLocale' ``` - or **remove value object and use simple types** ```yml rectors: Rector\Rector\Dynamic\ValueObjectRemoverRector: # type: new simple type 'ValueObjects\Name': 'string' ``` For example: ```php $value = new ValueObjects\Name('Tomas'); # to $value = 'Tomas'; ``` ```php /** * @var ValueObjects\Name */ private $name; # to /** * @var string */ private $name; ``` ```php public function someMethod(ValueObjects\Name $name) { ... # to public function someMethod(string $name) { ... ``` ### Turn Magic to Methods - **replace get/set magic methods with real ones** ```yml rectors: Rector\Rector\MagicDisclosure\GetAndSetToMethodCallRector: # class 'Nette\DI\Container': # magic method (prepared keys): new real method 'get': 'getService' 'set': 'addService' ``` For example: ```php $result = $container['key']; # to $result = $container->getService('key'); ``` ```php $container['key'] = $value; # to $container->addService('key', $value); ``` - or **replaces isset/unset magic methods with real ones** ```yml rectors: Rector\Rector\MagicDisclosure\UnsetAndIssetToMethodCallRector: # class 'Nette\DI\Container': # magic method (prepared keys): new real method 'isset': 'hasService' 'unset': 'removeService' ``` For example: ```php isset($container['key']); # to $container->hasService('key'); ``` ```php unset($container['key']); # to $container->removeService('key'); ``` - or **replaces toString magic methods with real ones** ```yml rectors: Rector\Rector\MagicDisclosure\ToStringToMethodCallRector: # class 'Symfony\Component\Config\ConfigCache': # magic method (prepared key): new real method 'toString': 'getPath' ``` For example: ```php $result = (string) $someValue; # to $result = $someValue->someMethod(); ``` ```php $result = $someValue->__toString(); # to $result = $someValue->someMethod(); ``` ### Coding Standards are Outsourced This package has no intention in formatting your code, as **coding standard tools handle this much better**. We prefer [EasyCodingStandard](https://github.com/Symplify/EasyCodingStandard) that is already available: ```bash # check vendor/bin/ecs check --config vendor/rector/rector/ecs-after-rector.neon # fix vendor/bin/ecs check --config vendor/rector/rector/ecs-after-rector.neon --fix ``` ### More Detailed Documentation - [How to Create Own Rector](/docs/HowToCreateOwnRector.md) - [Service Name to Type Provider](/docs/ServiceNameToTypeProvider.md) ### How to Contribute Just follow 3 rules: - **1 feature per pull-request** - **New feature needs tests** - Tests, coding standard and PHPStan **checks must pass** ```bash composer complete-check ``` Don you need to fix coding standards? Run: ```bash composer fix-cs ``` We would be happy to merge your feature then.