|
| 1 | +<!--suppress HtmlDeprecatedAttribute --> |
| 2 | +<h1 align="center">PHP Code Transformer</h1> |
| 3 | + |
| 4 | +<!-- Main Badges --> |
| 5 | +<p align="center"> |
| 6 | + <!-- License: MIT --> |
| 7 | + <a href="https://opensource.org/licenses/MIT" target="_blank"> |
| 8 | + <img alt="License: MIT" src="https://img.shields.io/badge/License-MIT-9C0000.svg?labelColor=ebdbb2&style=flat&logo="/> |
| 9 | + </a> |
| 10 | + |
| 11 | + <!-- Twitter: @WalterWoshid --> |
| 12 | + <a href="https://twitter.com/WalterWoshid" target="_blank"> |
| 13 | + <img alt="Twitter: @WalterWoshid" src="https://img.shields.io/badge/@WalterWoshid-Twitter?labelColor=ebdbb2&style=flat&logo=twitter&logoColor=458588&color=458588&label=Twitter"/> |
| 14 | + </a> |
| 15 | + |
| 16 | + <!-- PHP: >=8.1 --> |
| 17 | + <a href="https://www.php.net" target="_blank"> |
| 18 | + <img alt="PHP: >=8.1" src="https://img.shields.io/badge/PHP->=8.1-4C5789.svg?labelColor=ebdbb2&style=flat&logo=php&logoColor=4C5789"/> |
| 19 | + </a> |
| 20 | + |
| 21 | + <!-- Packagist --> |
| 22 | + <a href="https://packagist.org/packages/okapi/code-transformer" target="_blank"> |
| 23 | + <img alt="Packagist" src="https://img.shields.io/packagist/v/okapi/code-transformer?label=Packagist&labelColor=ebdbb2&style=flat&color=fe8019&logo=packagist"/> |
| 24 | + </a> |
| 25 | + |
| 26 | + <!-- Build --> |
| 27 | + <!--suppress HtmlUnknownTarget --> |
| 28 | + <a href="../../actions/workflows/tests.yml" target="_blank"> |
| 29 | + <img alt="Build" src="https://img.shields.io/github/actions/workflow/status/okapi-web/php-code-transformer/tests.yml?label=Build&labelColor=ebdbb2&style=flat&logo="> |
| 30 | + </a> |
| 31 | +</p> |
| 32 | + |
| 33 | +<!-- Coverage --> |
| 34 | +<p align="center"> |
| 35 | + <!-- Coverage - PHP 8.1 --> |
| 36 | + <a href="https://app.codecov.io/gh/okapi-web/php-code-transformer/flags" target="_blank"> |
| 37 | + <img alt="Coverage - PHP 8.1" src="https://img.shields.io/codecov/c/github/okapi-web/php-code-transformer?flag=os-ubuntu-latest_php-8.1&label=Coverage - PHP 8.1&labelColor=ebdbb2&style=flat&logo=codecov&logoColor=FFC107&color=FFC107"/> |
| 38 | + </a> |
| 39 | + |
| 40 | + <!-- Coverage - PHP 8.2 --> |
| 41 | + <a href="https://app.codecov.io/gh/okapi-web/php-code-transformer/flags" target="_blank"> |
| 42 | + <img alt="Coverage - PHP 8.2" src="https://img.shields.io/codecov/c/github/okapi-web/php-code-transformer?flag=os-ubuntu-latest_php-8.2&label=Coverage - PHP 8.2&labelColor=ebdbb2&style=flat&logo=codecov&logoColor=FFC107&color=FFC107"/> |
| 43 | + </a> |
| 44 | +</p> |
| 45 | + |
| 46 | +<h2 align="center">PHP Code Transformer is a PHP library that allows you to modify and transform the source code of a loaded PHP class.</h2> |
| 47 | + |
| 48 | + |
| 49 | + |
| 50 | +## Installation |
| 51 | + |
| 52 | +```shell |
| 53 | +composer require okapi/code-transformer |
| 54 | +``` |
| 55 | + |
| 56 | + |
| 57 | + |
| 58 | +# Usage |
| 59 | + |
| 60 | +## 📖 List of contents |
| 61 | + |
| 62 | +- [Create a kernel](#create-a-kernel) |
| 63 | +- [Create a transformer](#create-a-transformer) |
| 64 | +- [Initialize the kernel](#initialize-the-kernel) |
| 65 | +- [Result](#result) |
| 66 | + |
| 67 | + |
| 68 | + |
| 69 | +## Create a kernel |
| 70 | + |
| 71 | +```php |
| 72 | +<?php |
| 73 | + |
| 74 | +use Okapi\CodeTransformer\CodeTransformerKernel; |
| 75 | + |
| 76 | +class Kernel extends CodeTransformerKernel |
| 77 | +{ |
| 78 | + // Define a list of transformer classes |
| 79 | + protected array $transformers = [ |
| 80 | + StringTransformer::class, |
| 81 | + UnPrivateTransformer::class, |
| 82 | + ]; |
| 83 | +} |
| 84 | +``` |
| 85 | + |
| 86 | + |
| 87 | +## Create a transformer |
| 88 | + |
| 89 | +```php |
| 90 | +// String Transformer |
| 91 | + |
| 92 | +<?php |
| 93 | + |
| 94 | +use Okapi\CodeTransformer\Service\StreamFilter\Metadata\Code; |
| 95 | +use Okapi\CodeTransformer\Transformer; |
| 96 | + |
| 97 | +class StringTransformer extends Transformer |
| 98 | +{ |
| 99 | + public function getTargetClass(): string|array |
| 100 | + { |
| 101 | + // You can specify a single class or an array of classes |
| 102 | + // You can also use wildcards, see https://github.com/okapi-web/php-wildcards |
| 103 | + return MyTargetClass::class; |
| 104 | + } |
| 105 | + |
| 106 | + public function transform(Code $code): void |
| 107 | + { |
| 108 | + // Get the source file node which contains all the nodes of the source code |
| 109 | + $sourceFileNode = $code->sourceFileNode; |
| 110 | + |
| 111 | + // I recommend using the Microsoft\PhpParser library to parse the source code, |
| 112 | + // but you can also use any other library or manually parse the source code |
| 113 | + // or just use basic PHP functions with `$code->getOriginalSource()` |
| 114 | + |
| 115 | + // Iterate over all nodes |
| 116 | + foreach ($sourceFileNode->getDescendantNodes() as $node) { |
| 117 | + // Find 'Hello World!' string |
| 118 | + if ($node instanceof StringLiteral |
| 119 | + && $node->getStringContentsText() === 'Hello World!' |
| 120 | + ) { |
| 121 | + // Replace it with 'Hello from Code Transformer!' |
| 122 | + // Edit method accepts a Token class |
| 123 | + $code->edit( |
| 124 | + $node->children, |
| 125 | + "'Hello from Code Transformer!'", |
| 126 | + ); |
| 127 | + |
| 128 | + // You can also manually edit the source code |
| 129 | + $code->editAt( |
| 130 | + $node->getStartPosition(), |
| 131 | + $node->getWidth(), |
| 132 | + "'Hello from Code Transformer!'", |
| 133 | + ); |
| 134 | + |
| 135 | + // Append a new line of code |
| 136 | + $code->append('$iAmAppended = true;'); |
| 137 | + } |
| 138 | + } |
| 139 | + } |
| 140 | +} |
| 141 | +``` |
| 142 | + |
| 143 | +```php |
| 144 | +// UnPrivate Transformer |
| 145 | + |
| 146 | +<?php |
| 147 | + |
| 148 | +namespace Okapi\CodeTransformer\Tests\Stubs\Transformer; |
| 149 | + |
| 150 | +use Microsoft\PhpParser\TokenKind; |
| 151 | +use Okapi\CodeTransformer\Service\StreamFilter\Metadata\Code; |
| 152 | +use Okapi\CodeTransformer\Transformer; |
| 153 | + |
| 154 | +class UnPrivateTransformer extends Transformer |
| 155 | +{ |
| 156 | + public function getTargetClass(): string|array |
| 157 | + { |
| 158 | + return MyTargetClass::class; |
| 159 | + } |
| 160 | + |
| 161 | + public function transform(Code $code): void |
| 162 | + { |
| 163 | + $sourceFileNode = $code->sourceFileNode; |
| 164 | + |
| 165 | + // Iterate over all tokens |
| 166 | + foreach ($sourceFileNode->getDescendantTokens() as $token) { |
| 167 | + // Replace all private keywords with public |
| 168 | + if ($token->kind === TokenKind::PrivateKeyword) { |
| 169 | + $code->edit($token, 'public'); |
| 170 | + } |
| 171 | + } |
| 172 | + } |
| 173 | +} |
| 174 | +``` |
| 175 | + |
| 176 | + |
| 177 | +## Initialize the kernel |
| 178 | + |
| 179 | +```php |
| 180 | +// Initialize the kernel early in the application lifecycle |
| 181 | + |
| 182 | +<?php |
| 183 | + |
| 184 | +use MyKernel; |
| 185 | + |
| 186 | +require_once __DIR__ . '/vendor/autoload.php'; |
| 187 | + |
| 188 | +$kernel = new MyKernel( |
| 189 | + // The directory where the transformed source code will be stored |
| 190 | + cacheDir: __DIR__ . '/var/cache', |
| 191 | + |
| 192 | + // The cache file mode |
| 193 | + cacheFileMode: 0777, |
| 194 | +); |
| 195 | +``` |
| 196 | + |
| 197 | + |
| 198 | +## Result |
| 199 | + |
| 200 | +```php |
| 201 | +<?php |
| 202 | + |
| 203 | +// Just use your classes as usual |
| 204 | +$myTargetClass = new MyTargetClass(); |
| 205 | + |
| 206 | +$myTargetClass->myPrivateProperty; // You can't get me! |
| 207 | +$myTargetClass->myPrivateMethod(); // Hello from Code Transformer! |
| 208 | +``` |
| 209 | + |
| 210 | + |
| 211 | +```php |
| 212 | +// MyTargetClass.php |
| 213 | + |
| 214 | +<?php |
| 215 | + |
| 216 | +class MyTargetClass |
| 217 | +{ |
| 218 | + private string $myPrivateProperty = "You can't get me!"; |
| 219 | + |
| 220 | + private function myPrivateMethod(): void |
| 221 | + { |
| 222 | + echo 'Hello World!'; |
| 223 | + } |
| 224 | +} |
| 225 | +``` |
| 226 | + |
| 227 | +```php |
| 228 | +// MyTargetClass.php (transformed) |
| 229 | + |
| 230 | +<?php |
| 231 | + |
| 232 | +class MyTargetClass |
| 233 | +{ |
| 234 | + public string $myPrivateProperty = "You can't get me!"; |
| 235 | + |
| 236 | + public function myPrivateMethod(): void |
| 237 | + { |
| 238 | + echo 'Hello from Code Transformer!'; |
| 239 | + } |
| 240 | +} |
| 241 | +$iAmAppended = true; |
| 242 | +``` |
| 243 | + |
| 244 | + |
| 245 | +# How it works |
| 246 | + |
| 247 | +- The `Kernel` registers multiple services |
| 248 | + |
| 249 | + - The `TransformerContainer` service stores the list of transformers and their configuration |
| 250 | + |
| 251 | + - The `CacheStateManager` service manages the cache state |
| 252 | + |
| 253 | + - The `StreamFilter` service registers a [PHP Stream Filter](https://www.php.net/manual/wrappers.php.php#wrappers.php.filter) |
| 254 | + which allows to modify the source code before it is loaded by PHP |
| 255 | + |
| 256 | + - The `AutoloadInterceptor` service overloads the Composer autoloader, which handles the loading of classes |
| 257 | + |
| 258 | + |
| 259 | +## General workflow when a class is loaded |
| 260 | + |
| 261 | +- The `AutoloadInterceptor` service intercepts the loading of a class |
| 262 | + - It expects a class file path |
| 263 | + |
| 264 | +- The `TransformerContainer` matches the class name with the list of transformer target classes |
| 265 | + |
| 266 | +- If the class is matched, we query the cache state to see if the transformed source code is already cached |
| 267 | + - We check if the cache is valid |
| 268 | + - Modification time of the caching process is less than the modification time of the source file or the transformers |
| 269 | + - Check if the cache file, the source file and the transformers exist |
| 270 | + - If the cache is valid, we load the transformed source code from the cache |
| 271 | + - If not, we convert the class file path to a stream filter path |
| 272 | + |
| 273 | +- The `StreamFilter` modifies the source code by applying the matching transformers |
| 274 | + - If the modified source code is different from the original source code, we cache the transformed source code |
| 275 | + - If not, we cache it anyway, but without a cached source file path, so that the transformation process is not repeated |
| 276 | + |
| 277 | + |
| 278 | + |
| 279 | +## Testing |
| 280 | + |
| 281 | +- Run `composer run-script test`<br> |
| 282 | + or |
| 283 | +- Run `composer run-script test-coverage` |
| 284 | + |
| 285 | + |
| 286 | + |
| 287 | +## Show your support |
| 288 | + |
| 289 | +Give a ⭐ if this project helped you! |
| 290 | + |
| 291 | + |
| 292 | + |
| 293 | +## 📝 License |
| 294 | + |
| 295 | +Copyright © 2023 [Valentin Wotschel](https://github.com/WalterWoshid).<br> |
| 296 | +This project is [MIT](https://opensource.org/licenses/MIT) licensed. |
0 commit comments