Newsgroups: php.internals Path: news.php.net Xref: news.php.net php.internals:113544 Return-Path: Delivered-To: mailing list internals@lists.php.net Received: (qmail 66018 invoked from network); 15 Mar 2021 17:44:09 -0000 Received: from unknown (HELO php-smtp4.php.net) (45.112.84.5) by pb1.pair.com with SMTP; 15 Mar 2021 17:44:09 -0000 Received: from php-smtp4.php.net (localhost [127.0.0.1]) by php-smtp4.php.net (Postfix) with ESMTP id 51AF3180087 for ; Mon, 15 Mar 2021 10:37:42 -0700 (PDT) X-Spam-Checker-Version: SpamAssassin 3.4.2 (2018-09-13) on php-smtp4.php.net X-Spam-Level: X-Spam-Status: No, score=-2.6 required=5.0 tests=BAYES_00,NICE_REPLY_A, RCVD_IN_DNSWL_LOW,SPF_HELO_NONE,SPF_NONE autolearn=no autolearn_force=no version=3.4.2 X-Spam-Virus: No X-Envelope-From: Received: from wp160.webpack.hosteurope.de (wp160.webpack.hosteurope.de [80.237.132.167]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits) key-exchange ECDHE (P-256) server-signature RSA-PSS (4096 bits) server-digest SHA256) (No client certificate requested) by php-smtp4.php.net (Postfix) with ESMTPS for ; Mon, 15 Mar 2021 10:37:40 -0700 (PDT) Received: from [2a02:8109:9d40:1d44:40b6:f5d3:c567:d46c] (helo=nas.fritz.box); authenticated by wp160.webpack.hosteurope.de running ExIM with esmtpsa (TLS1.3:ECDHE_RSA_AES_128_GCM_SHA256:128) id 1lLr9u-0003vu-NM; Mon, 15 Mar 2021 18:37:38 +0100 To: internals@lists.php.net References: <2ff5377c-3718-feb1-ba20-96df47357414@mabe.berlin> Message-ID: <9f234d2a-7875-2871-6726-e7b8175099f3@mabe.berlin> Date: Mon, 15 Mar 2021 18:37:38 +0100 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:78.0) Gecko/20100101 Thunderbird/78.8.0 MIME-Version: 1.0 In-Reply-To: Content-Type: text/plain; charset=utf-8; format=flowed Content-Transfer-Encoding: 7bit Content-Language: en-US X-bounce-key: webpack.hosteurope.de;marc@mabe.berlin;1615829862;6d5bdfd0; X-HE-SMSGID: 1lLr9u-0003vu-NM Subject: Re: [PHP-DEV] Inconsistency in enumerations RFC From: marc@mabe.berlin (Marc) Hi Ilija - thanks for your answers On 14.03.21 23:18, Ilija Tovilo wrote: > Hi Marc > > Thanks for testing the enum feature. > >> Hi I'm trying out the new enumeration support for PHP 8.1 using >> https://github.com/php/php-src/pull/6489 thought to implement a polyfill >> based on class constants and reflection. Doing so I come across the >> newly defined interfaces: >> ``` interface UnitEnum{ >> public string$name; >> public staticfunction cases(): array ; >> } >> >> interface BackedEnumextends UnitEnum{ >> public string$value; >> public staticfunction from(int|string$scalar): static; >> public staticfunction tryFrom(int|string$scalar): ?static; >> } >> ``` My interest here is on BackedEnum which (currently) supports >> int|string. First of all the interface defines a "public string $value" >> but in fact the type needs to be "int|string". As properties are not >> supported in interfaces this is a pure documentation issue in the RFC >> itself. > We intentionally put it in the interface for clarity, even though it's > not technically possible yet. > >> Secondly the arguments name $scalar is not really a good name - >> now as we have named arguments - $value would be much more meaningful. > I think you might be looking at an older implementation. I only fixed > the naming recently (~2 weeks ago). The parameter is called $value, as > you suggested. nice +1 > >> But most interestingly I get the feeling that this should be split into >> several typed interfaces because this makes wrong impression especially >> that "tryFrom" would except int|string but actually it doesn't. >> Additionally, if we want to support other types later on this would blow >> up the types even more. Example: ``` enum Test:string { case TEST = '1'; >> } Test::tryFrom('unknown'); // NULL Test::tryFrom(1);// TypeError ``` >> The TypeError doesn't make sense here as it matches the type and it even >> matches the string representation of an existing type. > try/tryFrom will coerce the given value under non-strict_types. With > strict_types enabled it will type error. I think this behavior is > consistent with the rest of PHP. > >> I think it would >> be much clearer if there would be two different interfaces like: >> ``` interface IntEnum extends UnitEnum { public int $value; public >> static function from(int$value): static; public static function >> tryFrom(int$value): ?static; } >> interface StringEnum extends UnitEnum { public string $value; public >> static function from(string $value): static; public static function >> tryFrom(string $value): ?static; } ``` or else the above example should >> return a NULL or even case matching the string representation instead of >> TypeError as it matches the defined argument types. > I guess the interface would be more accurate. I'm not sure if that's a > huge win or just unnecessarily bloats the API. I think using > BackedEnum as a type hint should be relatively rare in practice. > Optimally we could add an associated type or generic type to > BackedEnum but that's currently not possible. I think it would be at least be helpful for example on mapping an enumeration from database to PHP as it's possible with doctrine types and probably also for static analyzers without the need to special case enumeration objects. I also don't think one additional interface bloats up the API or waiting for generics helps as nobody knows if this will ever be a thing in PHP. Would this actually require a follow-up RFC or could this be done as a small implementation detail change ? > Ilija >