Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix isValid static analysis #166

Closed
wants to merge 1 commit into from
Closed

Fix isValid static analysis #166

wants to merge 1 commit into from

Conversation

enumag
Copy link
Contributor

@enumag enumag commented Oct 27, 2022

I don't think the @psalm-assert-if-true is correct here. It causes false-positives like this with PHPStan 1.9:

Call to static method MyCLabs\Enum\Enum<string>::isValid() with string will always evaluate to true.

Which is of course wrong. Even if value is a string it doesn't necessarily mean it's one of the valid values for the enum.

Instead I think we should type $value with T since it doesn't really make sense to call isValid with a different type anyway - that would always return false.

cc @michaelpetri

@@ -209,9 +209,8 @@ public static function toArray()
* Check if is valid enum value
*
* @param $value
* @psalm-param mixed $value
* @psalm-param T $value

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

imo this doesn't make sense, if i declare value as T i would already knew it is valid, no?

Maybe you can rather prepare a fail case which proofs that there is a problem with the current implementation?

Copy link
Contributor Author

@enumag enumag Oct 28, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's assume this class:

/**
 * @extends Enum<string>
 */
class Status extends Enum {
    const SUCCESS = 'SUCCESS';
    const FAILURE = 'FAILURE';
}

The behavior of isValid is as follows:

var_dump(Status::isValid('SUCCESS')); // true
var_dump(Status::isValid('UNKNOWN')); // false
var_dump(Status::isValid(1)); // false

With your implementation static analysis fails for all 3:

// Call to static method MyCLabs\Enum\Enum<string>::isValid() with 'SUCCESS' will always evaluate to true. 
var_dump(Status::isValid('SUCCESS')); // true

// Call to static method MyCLabs\Enum\Enum<string>::isValid() with 'UNKNOWN' will always evaluate to true. 
var_dump(Status::isValid('UNKNOWN')); // false + the static analysis is incorrect

// Call to static method MyCLabs\Enum\Enum<string>::isValid() with 1 will always evaluate to false. 
var_dump(Status::isValid(1)); // false

With my suggestion:

var_dump(Status::isValid('SUCCESS')); // true, static analysis passes
var_dump(Status::isValid('UNKNOWN')); // false, static analysis passes
var_dump(Status::isValid(1)); // false, static analysis fails saying that isValid expects a string, int given

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

imo the problem is your Enum<string> annotation, with it you declare T as any string, but it is only SUCCESS and FAILURE.

Can your try this:

/**
 * @extends Enum<Status::*>
 */
class Status extends Enum {
    const SUCCESS = 'SUCCESS';
    const FAILURE = 'FAILURE';
}

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ooooh! Okay, interesting. I was completely unaware of that syntax. With that it works fine of course. It should be mentioned in the readme somewhere. Thanks!

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agree with the readme, maybe you can just open a PR for it?

In addition, worth to mention you can also use only a specific scope of constants

/**
 * @extends Enum<Status::VALUE_*>
 */
class Status extends Enum {
    const VALUE_SUCCESS = 'SUCCESS';
    const VALUE_FAILURE = 'FAILURE';
    const FOO = 'BAR';
}

or as literal values

/**
 * @extends Enum<'SUCCESS'|'FAILURE'>
 */
class Status extends Enum {
    const SUCCESS = 'SUCCESS';
    const FAILURE = 'FAILURE';
}

@enumag enumag closed this Oct 28, 2022
@enumag enumag mentioned this pull request Oct 28, 2022
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants