Find this useful? Enter your email to receive occasional updates for securing PHP code.
Signing you up...
Thank you for signing up!
PHP Decode
<?php declare(strict_types=1); namespace Larastan\Larastan\Rules; use Illuminate\Contra..
Decoded Output download
<?php
declare(strict_types=1);
namespace Larastan\Larastan\Rules;
use Illuminate\Contracts\Foundation\Application;
use PhpParser\Node;
use PhpParser\Node\Expr\MethodCall;
use PhpParser\NodeFinder;
use PHPStan\Analyser\Scope;
use PHPStan\Rules\Rule;
use PHPStan\Rules\RuleError;
use PHPStan\Rules\RuleErrorBuilder;
use PHPStan\Type\ObjectType;
use function array_map;
use function count;
use function in_array;
/** @implements Rule<MethodCall> */
class OctaneCompatibilityRule implements Rule
{
public function getNodeType(): string
{
return MethodCall::class;
}
/**
* @param MethodCall $node
*
* @return RuleError[]
*/
public function processNode(Node $node, Scope $scope): array
{
if (! $node->name instanceof Node\Identifier) {
return [];
}
if (! in_array($node->name->name, ['singleton', 'bind'], true)) {
return [];
}
$args = $node->getArgs();
if (count($args) < 2) {
return [];
}
$calledOnType = $scope->getType($node->var);
$classNames = $calledOnType->getObjectClassNames();
if (count($classNames) !== 1) {
return [];
}
if (
$classNames[0] !== Application::class &&
! (new ObjectType(Application::class))->isSuperTypeOf($calledOnType)->yes()
) {
return [];
}
if (! $args[1]->value instanceof Node\Expr\Closure) {
return [];
}
/** @var Node\Expr\Closure $closure */
$closure = $args[1]->value;
/** @var Node\Param[] $closureParams */
$closureParams = $closure->getParams();
// Closure should have at least one parameter. First param
// is container, second is parameters. If no parameter
// is given we will check for the usage of `$this->app`
if (count($closureParams) < 1) {
return $this->checkForThisAppUsage($scope, $closure);
}
// Using `$app` with `bind` is ok, so we return early
if ($node->name->name === 'bind') {
return [];
}
if (! $closureParams[0]->var instanceof Node\Expr\Variable) {
return [];
}
$containerParameterName = $closureParams[0]->var->name;
$nodes = (new NodeFinder())->find($closure->getStmts(), static function (Node $node) use ($containerParameterName): bool {
if (! $node instanceof Node\Expr\New_) {
return false;
}
if (count($node->getArgs()) < 1) {
return false;
}
if (! $node->getArgs()[0]->value instanceof Node\Expr\Variable && ! $node->getArgs()[0]->value instanceof Node\Expr\ArrayDimFetch) {
return false;
}
if ($node->getArgs()[0]->value instanceof Node\Expr\ArrayDimFetch) {
/** @var Node\Expr\Variable $var */
$var = $node->getArgs()[0]->value->var;
if ($var->name !== $containerParameterName) {
return false;
}
if ($node->getArgs()[0]->value->dim === null) {
return false;
}
if (! $node->getArgs()[0]->value->dim instanceof Node\Scalar\String_) {
return false;
}
return in_array($node->getArgs()[0]->value->dim->value, ['request', 'config'], true);
}
return $node->getArgs()[0]->value->name === $containerParameterName;
});
if (count($nodes) > 0) {
return array_map([$this, 'dependencyInjectionError'], $nodes);
}
return [];
}
/** @return RuleError[] */
private function checkForThisAppUsage(Scope $scope, Node\Expr\Closure $closure): array
{
$nodes = (new NodeFinder())->find($closure->getStmts(), static function (Node $node): bool {
return $node instanceof Node\Expr\PropertyFetch &&
$node->var instanceof Node\Expr\Variable &&
$node->var->name === 'this' &&
$node->name instanceof Node\Identifier &&
$node->name->name === 'app';
});
if (count($nodes) > 0) {
return array_map([$this, 'dependencyInjectionError'], $nodes);
}
return [];
}
private function dependencyInjectionError(Node $node): RuleError
{
return RuleErrorBuilder::message('Consider using bind method instead or pass a closure.')
->identifier('larastan.octaneCompatibility')
->tip('See: https://laravel.com/docs/octane#dependency-injection-and-octane')
->line($node->getAttribute('startLine'))
->build();
}
}
?>
Did this file decode correctly?
Original Code
<?php
declare(strict_types=1);
namespace Larastan\Larastan\Rules;
use Illuminate\Contracts\Foundation\Application;
use PhpParser\Node;
use PhpParser\Node\Expr\MethodCall;
use PhpParser\NodeFinder;
use PHPStan\Analyser\Scope;
use PHPStan\Rules\Rule;
use PHPStan\Rules\RuleError;
use PHPStan\Rules\RuleErrorBuilder;
use PHPStan\Type\ObjectType;
use function array_map;
use function count;
use function in_array;
/** @implements Rule<MethodCall> */
class OctaneCompatibilityRule implements Rule
{
public function getNodeType(): string
{
return MethodCall::class;
}
/**
* @param MethodCall $node
*
* @return RuleError[]
*/
public function processNode(Node $node, Scope $scope): array
{
if (! $node->name instanceof Node\Identifier) {
return [];
}
if (! in_array($node->name->name, ['singleton', 'bind'], true)) {
return [];
}
$args = $node->getArgs();
if (count($args) < 2) {
return [];
}
$calledOnType = $scope->getType($node->var);
$classNames = $calledOnType->getObjectClassNames();
if (count($classNames) !== 1) {
return [];
}
if (
$classNames[0] !== Application::class &&
! (new ObjectType(Application::class))->isSuperTypeOf($calledOnType)->yes()
) {
return [];
}
if (! $args[1]->value instanceof Node\Expr\Closure) {
return [];
}
/** @var Node\Expr\Closure $closure */
$closure = $args[1]->value;
/** @var Node\Param[] $closureParams */
$closureParams = $closure->getParams();
// Closure should have at least one parameter. First param
// is container, second is parameters. If no parameter
// is given we will check for the usage of `$this->app`
if (count($closureParams) < 1) {
return $this->checkForThisAppUsage($scope, $closure);
}
// Using `$app` with `bind` is ok, so we return early
if ($node->name->name === 'bind') {
return [];
}
if (! $closureParams[0]->var instanceof Node\Expr\Variable) {
return [];
}
$containerParameterName = $closureParams[0]->var->name;
$nodes = (new NodeFinder())->find($closure->getStmts(), static function (Node $node) use ($containerParameterName): bool {
if (! $node instanceof Node\Expr\New_) {
return false;
}
if (count($node->getArgs()) < 1) {
return false;
}
if (! $node->getArgs()[0]->value instanceof Node\Expr\Variable && ! $node->getArgs()[0]->value instanceof Node\Expr\ArrayDimFetch) {
return false;
}
if ($node->getArgs()[0]->value instanceof Node\Expr\ArrayDimFetch) {
/** @var Node\Expr\Variable $var */
$var = $node->getArgs()[0]->value->var;
if ($var->name !== $containerParameterName) {
return false;
}
if ($node->getArgs()[0]->value->dim === null) {
return false;
}
if (! $node->getArgs()[0]->value->dim instanceof Node\Scalar\String_) {
return false;
}
return in_array($node->getArgs()[0]->value->dim->value, ['request', 'config'], true);
}
return $node->getArgs()[0]->value->name === $containerParameterName;
});
if (count($nodes) > 0) {
return array_map([$this, 'dependencyInjectionError'], $nodes);
}
return [];
}
/** @return RuleError[] */
private function checkForThisAppUsage(Scope $scope, Node\Expr\Closure $closure): array
{
$nodes = (new NodeFinder())->find($closure->getStmts(), static function (Node $node): bool {
return $node instanceof Node\Expr\PropertyFetch &&
$node->var instanceof Node\Expr\Variable &&
$node->var->name === 'this' &&
$node->name instanceof Node\Identifier &&
$node->name->name === 'app';
});
if (count($nodes) > 0) {
return array_map([$this, 'dependencyInjectionError'], $nodes);
}
return [];
}
private function dependencyInjectionError(Node $node): RuleError
{
return RuleErrorBuilder::message('Consider using bind method instead or pass a closure.')
->identifier('larastan.octaneCompatibility')
->tip('See: https://laravel.com/docs/octane#dependency-injection-and-octane')
->line($node->getAttribute('startLine'))
->build();
}
}
Function Calls
None |
Stats
MD5 | 1def41cab58a336f2211ca410cf8089d |
Eval Count | 0 |
Decode Time | 86 ms |