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

[RFC] Optional interfaces #17288

Closed
wants to merge 25 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
4358a05
Add syntax for optional interfaces
tontonsb Dec 28, 2024
a0fe072
Remove resolved comments
tontonsb Dec 29, 2024
4ed9176
Move the `num_interfaces` mutation to a safer place
tontonsb Dec 29, 2024
eba9a20
Test optional interfaces
tontonsb Feb 9, 2025
1386ca2
Fix memory leak
tontonsb Feb 9, 2025
686d2e9
Add test for Stringable with optional interfaces
tontonsb Feb 9, 2025
b00e2f3
Simplify test
tontonsb Feb 9, 2025
99f0837
Make test line-trimming-resilient
tontonsb Feb 9, 2025
48ab59a
Test Optional interfaces on OPCache
tontonsb Feb 9, 2025
8a90459
Specify conflict on an OPCache test
tontonsb Feb 9, 2025
fd190aa
Generate test script
tontonsb Feb 9, 2025
44f1a63
Enable OPCache in the optional interface test
tontonsb Feb 9, 2025
77f5ebc
Fix permissions
tontonsb Feb 9, 2025
9efccd2
Update Zend/zend_compile.c
tontonsb Feb 14, 2025
a64d1e9
Clean up
tontonsb Feb 14, 2025
35a5491
Make const naming more precise
tontonsb Feb 14, 2025
c3efd97
Move efree back
tontonsb Feb 14, 2025
d216c35
Make name qualifying checks safe
tontonsb Feb 14, 2025
de36608
Fix mask
tontonsb Feb 14, 2025
74b524f
[skip ci] Fix spacing
iluuu1994 Feb 18, 2025
2f8405c
Fix ZEND_NAME_ for new class name in const expr
iluuu1994 Feb 18, 2025
62171fd
Fix leaking of interface_names when interface count goes to 0
iluuu1994 Feb 18, 2025
3e461b8
Fix shm modification
iluuu1994 Feb 18, 2025
821ec0f
Fix optional interface test on windows
tontonsb Feb 20, 2025
b7e513e
Rework optional interface opcache test
tontonsb Feb 21, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
24 changes: 24 additions & 0 deletions Zend/tests/optional_interfaces/autoloading.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
--TEST--
Optional interfaces are autoloaded
--FILE--
<?php

spl_autoload_register(function ($class) {
echo "Autoloading: $class\n";

if ($class === 'ExistingInterface')
eval("interface ExistingInterface {}");
});

class TestClass implements ?ExistingInterface, ?NonExistantInterface {}

$c = new TestClass;
echo implode(',', class_implements($c))."\n";
echo implode(',', class_implements('TestClass'))."\n";

?>
--EXPECT--
Autoloading: ExistingInterface
Autoloading: NonExistantInterface
ExistingInterface
ExistingInterface
41 changes: 41 additions & 0 deletions Zend/tests/optional_interfaces/dynamic_definition.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
--TEST--
Dynamically defined interfaces only affect classes defined later
--FILE--
<?php

class AncientClass implements ?OptionalInterface {}
$ancientClass = new AncientClass;
echo 'AC implements:'.implode(', ', class_implements('AncientClass')),"\n";
echo 'AC object implements:'.implode(', ', class_implements($ancientClass)),"\n\n";

eval('interface OptionalInterface {}');
echo "Interface defined\n\n";

class NewClass implements ?OptionalInterface {}

// Newly defined class implements the interface
$newClass = new NewClass;
echo 'NC implements:'.implode(', ', class_implements('NewClass')),"\n";
echo 'NC implements:'.implode(', ', class_implements($newClass)),"\n\n";

// The old class doesn't
echo 'AC object implements:'.implode(', ', class_implements($ancientClass)),"\n";

echo 'AC implements:'.implode(', ', class_implements('AncientClass')),"\n";

$ancientClass2 = new AncientClass;
echo 'New AC object implements:'.implode(', ', class_implements($ancientClass2)),"\n";

?>
--EXPECT--
AC implements:
AC object implements:

Interface defined

NC implements:OptionalInterface
NC implements:OptionalInterface

AC object implements:
AC implements:
New AC object implements:
13 changes: 13 additions & 0 deletions Zend/tests/optional_interfaces/optional_interface_with_parent.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
--TEST--
Optional interface with parent
--FILE--
<?php

interface I {}
class P implements I {}
class C extends P implements ?OptionalInterface {}

?>
===DONE===
--EXPECT--
===DONE===
77 changes: 77 additions & 0 deletions Zend/tests/optional_interfaces/optional_interfaces.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
--TEST--
Optional interfaces work
--FILE--
<?php

interface ExistingInterface {}

class ImplementingOptionalInterface implements ?ExistingInterface {}
class SkippingOptionalInterface implements ?NonExistantInterface {}
class MultipleOptionalsExistingFirst implements ?ExistingInterface, ?NonExistantInterface {}
class MultipleOptionalsExistingLast implements ?NonExistantInterface, ?ExistingInterface {}
class MixedOptionalFirst implements ?NonExistantInterface, ExistingInterface {}
class MixedOptionalLast implements ExistingInterface, ?NonExistantInterface {}

interface ExtendingExistingOptional extends ExistingInterface {}
interface ExtendingNonexistantOptional extends ?NonExistantInterface {}

class ImplementsInheritedExisting implements ExtendingExistingOptional {}
class ImplementsInheritedSkipped implements ExtendingNonexistantOptional {}

$classes = [
ImplementingOptionalInterface::class,
SkippingOptionalInterface::class,
MultipleOptionalsExistingFirst::class,
MultipleOptionalsExistingLast::class,
MixedOptionalFirst::class,
MixedOptionalLast::class,
ImplementsInheritedExisting::class,
ImplementsInheritedSkipped::class,
];

foreach ($classes as $class) {
echo "$class\n";
echo ' class implements:'.implode(', ', class_implements($class))."\n";
echo ' object implements:'.implode(', ', class_implements(new $class))."\n";
}

$interfaces = [
ExtendingExistingOptional::class,
ExtendingNonexistantOptional::class,
];

foreach ($interfaces as $interface) {
echo "$interface\n";
echo ' interface extends:'.implode(', ', class_implements($class))."\n";
}

?>
--EXPECT--
ImplementingOptionalInterface
class implements:ExistingInterface
object implements:ExistingInterface
SkippingOptionalInterface
class implements:
object implements:
MultipleOptionalsExistingFirst
class implements:ExistingInterface
object implements:ExistingInterface
MultipleOptionalsExistingLast
class implements:ExistingInterface
object implements:ExistingInterface
MixedOptionalFirst
class implements:ExistingInterface
object implements:ExistingInterface
MixedOptionalLast
class implements:ExistingInterface
object implements:ExistingInterface
ImplementsInheritedExisting
class implements:ExtendingExistingOptional, ExistingInterface
object implements:ExtendingExistingOptional, ExistingInterface
ImplementsInheritedSkipped
class implements:ExtendingNonexistantOptional
object implements:ExtendingNonexistantOptional
ExtendingExistingOptional
interface extends:ExtendingNonexistantOptional
ExtendingNonexistantOptional
interface extends:ExtendingNonexistantOptional
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
--TEST--
Optional interfaces are checked
--FILE--
<?php

interface Iface
{
public function method();
}

class DecentClass implements ?Iface
{
public function method() {}
}

$anInstance = new DecentClass;
if ($anInstance instanceof Iface)
echo "Existing interfaces can be implemented.";

class BadClass implements ?Iface {}

?>
--EXPECTF--
Existing interfaces can be implemented.
Fatal error: Class BadClass contains 1 abstract method and must therefore be declared abstract or implement the remaining method (Iface::method) in %soptional_interfaces_are_checked.php on line %d
22 changes: 22 additions & 0 deletions Zend/tests/optional_interfaces/override.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
--TEST--
Optional interfaces doesn't make Override optional
--FILE--
<?php

interface ExistingInterface
{
public function method();
}

class TestClass implements ?ExistingInterface, ?NonExistantInterface
{
#[\Override]
public function method() {}

#[\Override]
public function other() {}
}

?>
--EXPECTF--
Fatal error: TestClass::other() has #[\Override] attribute, but no matching parent method exists in %soverride.php on line %d
15 changes: 15 additions & 0 deletions Zend/tests/optional_interfaces/reflection.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
--TEST--
Reflection works on optional interfaces
--FILE--
<?php

interface ExistingInterface {}

class TestClass implements ?ExistingInterface, ?NonexistantInterface {}

$reflection = new ReflectionClass('TestClass');
echo implode(', ', $reflection->getInterfaceNames());

?>
--EXPECT--
ExistingInterface
16 changes: 16 additions & 0 deletions Zend/tests/optional_interfaces/serialization.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
--TEST--
Classes with optional interfaces survive serialization
--FILE--
<?php

interface ExistingInterface {}
class TestClass implements ?ExistingInterface, ?NonExistantInterface {}

$original = new TestClass;
$deserialized = unserialize(serialize($original));

echo implode(',', class_implements($deserialized));

?>
--EXPECT--
ExistingInterface
20 changes: 20 additions & 0 deletions Zend/tests/optional_interfaces/stringable.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
--TEST--
Adding stringable is not broken with Optional interfaces
--FILE--
<?php

trait Str
{
public function __toString() {}
}

class TestClass implements ?NonexistantInterface
{
use Str;
}

echo implode(',', class_implements('TestClass'));

?>
--EXPECT--
Stringable
27 changes: 27 additions & 0 deletions Zend/tests/optional_interfaces/type_checks.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
--TEST--
Only the existing interfaces pass the type checks
--INI--
zend.exception_ignore_args = On
Copy link
Member

Choose a reason for hiding this comment

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

Nit: We don't use spaces in the INI section.

--FILE--
<?php

interface ExistingInterface {}

class TestClass implements ?ExistingInterface, ?NonExistantInterface {}

function f1(ExistingInterface $x) { echo "F1"; }
function f2(NonExistantInterface $x) { echo "F2"; }

$c = new TestClass;

f1($c);
f2($c);

?>
--EXPECTF--
F1
Fatal error: Uncaught TypeError: f2(): Argument #1 ($x) must be of type NonExistantInterface, TestClass given, called in %stype_checks.php on line %d and defined in %stype_checks.php:%d
Stack trace:
#0 %s
#1 {main}
thrown in %stype_checks.php on line %d
8 changes: 7 additions & 1 deletion Zend/zend.h
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,12 @@ typedef struct _zend_class_name {
zend_string *lc_name;
} zend_class_name;

typedef struct _zend_interface_name {
zend_string *name;
zend_string *lc_name;
bool is_optional;
} zend_interface_name;

typedef struct _zend_trait_method_reference {
zend_string *method_name;
zend_string *class_name;
Expand Down Expand Up @@ -210,7 +216,7 @@ struct _zend_class_entry {
/* class_entry or string(s) depending on ZEND_ACC_LINKED */
union {
zend_class_entry **interfaces;
zend_class_name *interface_names;
zend_interface_name *interface_names;
};

zend_class_name *trait_names;
Expand Down
6 changes: 3 additions & 3 deletions Zend/zend_ast.c
Original file line number Diff line number Diff line change
Expand Up @@ -1394,11 +1394,11 @@ static ZEND_COLD void zend_ast_export_ns_name(smart_str *str, zend_ast *ast, int
zval *zv = zend_ast_get_zval(ast);

if (Z_TYPE_P(zv) == IS_STRING) {
if (ast->attr == ZEND_NAME_FQ) {
if (NAME_QUAL(ast->attr) == ZEND_NAME_FQ) {
smart_str_appendc(str, '\\');
} else if (ast->attr == ZEND_NAME_RELATIVE) {
} else if (NAME_QUAL(ast->attr) == ZEND_NAME_RELATIVE) {
smart_str_appends(str, "namespace\\");
}
}
smart_str_append(str, Z_STR_P(zv));
return;
}
Expand Down
Loading
Loading