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

WeakMap::offsetExists return false if value is null #8437

Open
julien-boudry opened this issue Apr 25, 2022 · 11 comments
Open

WeakMap::offsetExists return false if value is null #8437

julien-boudry opened this issue Apr 25, 2022 · 11 comments

Comments

@julien-boudry
Copy link

julien-boudry commented Apr 25, 2022

Description

The following code:

<?php

$wm = new \WeakMap;
$os = new \SplObjectStorage;

$obj1 = new \stdClass;
$obj2 = new \stdClass;

$wm[$obj1] = null;
$wm[$obj2] = true;

$os[$obj1] = null;
$os[$obj2] = true;


var_dump($wm->offsetExists($obj1));
var_dump($wm->offsetExists($obj2));

var_dump($os->offsetExists($obj1));
var_dump($os->offsetExists($obj2));


foreach ($wm as $key => $value) {
    if ($key === $obj1) {
        echo 'key $obj1 is always here';
    }
}

Resulted in this output:

bool(false)
bool(true)
bool(true)
bool(true)
key $obj1 is always here

But I expected this output instead:

bool(true)
bool(true)
bool(true)
bool(true)
key $obj1 is always here

PHP Version

PHP 8.1.5

Operating System

@cmb69
Copy link
Member

cmb69 commented Apr 25, 2022

return Z_TYPE_P(zv) != IS_NULL;

I wonder if that is deliberate or if s/IS_NULL/IS_UNDEF would be more appropriate.

@nikic
Copy link
Member

nikic commented Apr 26, 2022

This is deliberate. Despite the confusing name, the API contract of offsetExists is that it must return false for null values. It is the analogue of __isset() and must match isset() semantics.

@julien-boudry
Copy link
Author

This is deliberate. Despite the confusing name, the API contract of offsetExists is that it must return false for null values. It is the analogue of __isset() and must match isset() semantics.

Will other Spl implementations of ArrayAccess, like SplObjectStorage, will change in the future according to this?

@julien-boudry
Copy link
Author

julien-boudry commented Apr 26, 2022

  • Note that, in this example, var_dump( isset($os[$obj1]) ); return true.

@cmb69
Copy link
Member

cmb69 commented Apr 26, 2022

Despite the confusing name, the API contract of offsetExists is that it must return false for null values. It is the analogue of __isset() and must match isset() semantics.

Ah, right.

Will other Spl implementations of ArrayAccess, like SplObjectStorage, will change in the future according to this?

SplObjectStorage::offsetExists() is an alias of ::contains(), what is apparently a deliberate design decision:

/* NOTE: SplObjectStorage->offsetExists() is an alias of SplObjectStorage->contains(), so this returns true even if the value is null. */

I think we should clarify the respective note in the manual.

Are there other classes, where ::offsetExists() exhibits this behavior?

@julien-boudry
Copy link
Author

I think we should clarify the respective note in the manual.

Also on ArrayAccess manual, is unclear ( https://www.php.net/manual/en/arrayaccess.offsetexists.php ) even if it's talking about the empty() function

@MorganLOCode
Copy link

MorganLOCode commented Apr 17, 2023

Just encountered this myself (gettype($splos[$i]) is NULL (actually, never assigned to) but isset($splos[$i]) is true), and found nothing to suggest the behaviour was correct.

The only part of the manual I found that was relevant was the statement on the isset manual page that it returns false for a variable that has been assigned null.

Is it correct? If it's a deliberate decision, what was the rationale? Or is it just a emergent consequence of offsetExists/contains being aliased together?

@damianwadley
Copy link
Member

Just encountered this myself (gettype($splos[$i]) is NULL (actually, never assigned to) but isset($splot[$i]) is true), and found nothing to suggest the behaviour was correct.

I can't reproduce this. https://3v4l.org/NKKhE

Additionally, "never assigned to" suggests you didn't create $i in the $splot WeakMap, in which case gettype-ing it will throw an exception because it doesn't exist, so I'm not sure what you mean by that.

Unless, of course, the splos/splot typo in your comment is also present in your code?

@MorganLOCode
Copy link

MorganLOCode commented Apr 17, 2023

Try using SplObjectStorage instead of WeakMap per the comment from @cmb69 above. The issue seems more generally around the behaviour of offsetExists. I'll fix that typo.

In perhaps related behaviour, isset($storage[$unsetvar]) will spit warnings when $unsetvar is not initialised; that case is not dealt with before passing it off to the offsetExists method. Would moving unset/null tests up so they're done ahead of calling SPL methods be a way of addressing these in a general sense?

@iluuu1994
Copy link
Member

I had a quick look at that. Unfortunately, there attach method deliberately uses NULL as an empty value. So, changing the semantics to match isset means we'll also need to change this empty value. There's also this comment:

/* NOTE: SplObjectStorage->offsetExists() is an alias of SplObjectStorage->contains(), so this returns true even if the value is null. */

So this seems intentional, although dubious.

@Girgias Girgias self-assigned this Mar 8, 2024
@MorganLOCode
Copy link

In perhaps related behaviour, isset($storage[$unsetvar]) will spit warnings when $unsetvar is not initialised

Not sure why I thought this was a problem when I wrote that: it's always been normal behaviour. The test is whether the indexed element exists or not - which assumes the index exists.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

7 participants