self is self.id_data is inconsistent. #86170

Closed
opened 2021-03-01 22:15:00 +01:00 by Will · 11 comments

System Information
Operating system: Linux
Graphics card: Intel

Blender Version
Broken: 2.91.0
Worked: never

Short description of error

When the driver code self is self.id_data is evaluated on an Object property (E.G. Transform) in a file that contains in multiple drivers, it will sometimes return True and sometimes return False, for different updates on the same driver channel.

image.png

image.png

#Test.py
import bpy

bpy.app.driver_namespace["Test"] = lambda self: self is self.id_data

image.png

image.png

I'm not sure whether this is a bug, as the behaviour seems somewhat reasonable given the strictness of identity comparison and the complexity of Blender's API— The object might get moved around in memory or destroyed and re-made or something.

But on the other hand, I also think it seems reasonable to expect a single code line, executed sequentially, to behave consistently in that regard.

I think I was also confused because the behaviour seems much more consistent when the number of drivers is fewer and their relationship is simpler, and because this behaviour doesn't seem to occur in the interactive console (E.G. all(C.active_object is C.active_object.id_data for i in range(10000)) == True). So when testing the expression to figure out whether it's safe to use, I easily concluded that it seems fine, only to have problems once the scene was complex enough for it to be harder to track down the cause.

Also, if you assign self and self.id_data to variables before comparing them, and you print out the variables, then it will actually show that do in fact they have the same memory address (E.G. a, b = self, self.id_data; c = a is b; print(a, b, c) can produce <bpy_struct, Object("ObjectName") at 0x7f4b07a76208, evaluated> <bpy_struct, Object("ObjectName") at 0x7f4b07a76208, evaluated> False, where both Python objects are apparently at 0x7f4b07a76208.)

Exact steps for others to reproduce the error
Open the attached .blend file, then rapidly click "Update Dependencies" on a transform channel's driver until the Cube's location starts jumping around.

SelfIsntNotSelfIDData.blend

**System Information** Operating system: Linux Graphics card: Intel **Blender Version** Broken: 2.91.0 Worked: never **Short description of error** When the driver code `self is self.id_data` is evaluated on an Object property (E.G. Transform) in a file that contains in multiple drivers, it will sometimes return `True` and sometimes return `False`, for different updates on the same driver channel. ![image.png](https://archive.blender.org/developer/F9857835/image.png) ![image.png](https://archive.blender.org/developer/F9857837/image.png) ``` #Test.py import bpy bpy.app.driver_namespace["Test"] = lambda self: self is self.id_data ``` ![image.png](https://archive.blender.org/developer/F9857839/image.png) ![image.png](https://archive.blender.org/developer/F9857841/image.png) I'm not sure whether this is a bug, as the behaviour seems somewhat reasonable given the strictness of identity comparison and the complexity of Blender's API— The object might get moved around in memory or destroyed and re-made or something. But on the other hand, I also think it seems reasonable to expect a single code line, executed sequentially, to behave consistently in that regard. I think I was also confused because the behaviour seems much more consistent when the number of drivers is fewer and their relationship is simpler, and because this behaviour doesn't seem to occur in the interactive console (E.G. `all(C.active_object is C.active_object.id_data for i in range(10000)) == True`). So when testing the expression to figure out whether it's safe to use, I easily concluded that it seems fine, only to have problems once the scene was complex enough for it to be harder to track down the cause. Also, if you assign `self` and `self.id_data` to variables before comparing them, and you print out the variables, then it will actually show that do in fact they have the same memory address (E.G. `a, b = self, self.id_data; c = a is b; print(a, b, c)` can produce `<bpy_struct, Object("ObjectName") at 0x7f4b07a76208, evaluated> <bpy_struct, Object("ObjectName") at 0x7f4b07a76208, evaluated> False`, where both Python objects are apparently at `0x7f4b07a76208`.) **Exact steps for others to reproduce the error** Open the attached `.blend` file, then rapidly click "Update Dependencies" on a transform channel's driver until the `Cube`'s location starts jumping around. [SelfIsntNotSelfIDData.blend](https://archive.blender.org/developer/F9857867/SelfIsntNotSelfIDData.blend)
Author

Added subscriber: @WCN

Added subscriber: @WCN
Member

Added subscriber: @Mets

Added subscriber: @Mets
Member

From the PyAPI:
id_data: The bpy.types.ID object this datablock is from or None, (not available for all data types)

So, on a driver placed on a Constraint property, where self refers to the constraint, self.id_data will refer to the bpy.types.Object that holds the constraint. self != self.id_data.
On a driver placed on an Object property, where self refers to the object, self==self.id_data.

That said, I find the nomenclature around what's an "ID" and what's a "datablock" very confusing, so I would be interested to hear some developer input on that, but I don't think there is a bug here.

From the PyAPI: `id_data: The bpy.types.ID object this datablock is from or None, (not available for all data types)` So, on a driver placed on a Constraint property, where `self` refers to the constraint, `self.id_data` will refer to the bpy.types.Object that holds the constraint. self != self.id_data. On a driver placed on an Object property, where `self` refers to the object, self==self.id_data. That said, I find the nomenclature around what's an "ID" and what's a "datablock" very confusing, so I would be interested to hear some developer input on that, but I don't think there is a bug here.
Author

In #86170#1122225, @Mets wrote:
From the PyAPI:
id_data: The bpy.types.ID object this datablock is from or None, (not available for all data types)

So, on a driver placed on a Constraint property, where self refers to the constraint, self.id_data will refer to the bpy.types.Object that holds the constraint. self != self.id_data.
On a driver placed on an Object property, where self refers to the object, self==self.id_data.

Oh yeah, I'm aware of that. Sorry, I should have been clearer in my original post.

The inconsistency occurs when the driver is placed on the Object datablock. As you can see from my screenshots, self is self.id_data will sometimes be True and sometimes be False on drivers for the .location- [ ] property of the object— It's the same driver on the same Object property that evaluates differently every couple updates, even though both sides always refer to the same Blender Object() with apparently the same memory address.

It's a non-issue on Constraints, Modifiers, etc. because, as you note, there it will always return False anyway. Also, I should note that only is is inconsistent (and presumably is not is as well); == seems to work fine as I can tell. (But == is a magic method while is is a language feature, so especially since the memory addresses and str() of self and self.id_data seem to be the same even where is returns False and since is appears to work perfectly in all situations where a user could reasonably test it, there may be an argument for reasonably using is instead of ==.)

> In #86170#1122225, @Mets wrote: > From the PyAPI: > `id_data: The bpy.types.ID object this datablock is from or None, (not available for all data types)` > > So, on a driver placed on a Constraint property, where `self` refers to the constraint, `self.id_data` will refer to the bpy.types.Object that holds the constraint. self != self.id_data. > On a driver placed on an Object property, where `self` refers to the object, self==self.id_data. Oh yeah, I'm aware of that. Sorry, I should have been clearer in my original post. The inconsistency occurs when the driver is placed on the `Object` datablock. As you can see from my screenshots, `self is self.id_data` will sometimes be `True` and sometimes be `False` on drivers for the `.location- [ ]` property of the object— It's the same driver on the same `Object` property that evaluates differently every couple updates, even though both sides always refer to the same Blender `Object()` with apparently the same memory address. It's a non-issue on Constraints, Modifiers, etc. because, as you note, there it will always return `False` anyway. Also, I should note that only `is` is inconsistent (and presumably `is not` is as well); `==` seems to work fine as I can tell. (But `==` is a magic method while `is` is a language feature, so especially since the memory addresses and `str()` of `self` and `self.id_data` seem to be the same even where `is` returns `False` and since `is` appears to work perfectly in all situations where a user could reasonably test it, there may be an argument for reasonably using `is` instead of `==`.)
Member

Oh I see, thanks for the clarification! That does sound like a bug indeed :)

Oh I see, thanks for the clarification! That does sound like a bug indeed :)
Member

Added subscriber: @lichtwerk

Added subscriber: @lichtwerk
Member

Changed status from 'Needs Triage' to: 'Confirmed'

Changed status from 'Needs Triage' to: 'Confirmed'
Member

Thx for the detailed report.

Also, I should note that only is is inconsistent (and presumably is not is as well); == seems to work fine as I can tell.

For me == can also fail (although apparently less often.

Thx for the detailed report. > Also, I should note that only is is inconsistent (and presumably is not is as well); == seems to work fine as I can tell. For me `==` can also fail (although apparently less often.
Campbell Barton self-assigned this 2021-04-22 10:45:00 +02:00

This issue was referenced by 47cc05471d

This issue was referenced by 47cc05471d1c046ad89b7e60f52157ca7dcda075

This issue was referenced by a23e49c696

This issue was referenced by a23e49c696cf9ac66ecb9c1ad009d38b0a5e9f9f

Changed status from 'Confirmed' to: 'Resolved'

Changed status from 'Confirmed' to: 'Resolved'
Sign in to join this conversation.
No Label
Interest
Alembic
Interest
Animation & Rigging
Interest
Asset Browser
Interest
Asset Browser Project Overview
Interest
Audio
Interest
Automated Testing
Interest
Blender Asset Bundle
Interest
BlendFile
Interest
Collada
Interest
Compatibility
Interest
Compositing
Interest
Core
Interest
Cycles
Interest
Dependency Graph
Interest
Development Management
Interest
EEVEE
Interest
EEVEE & Viewport
Interest
Freestyle
Interest
Geometry Nodes
Interest
Grease Pencil
Interest
ID Management
Interest
Images & Movies
Interest
Import Export
Interest
Line Art
Interest
Masking
Interest
Metal
Interest
Modeling
Interest
Modifiers
Interest
Motion Tracking
Interest
Nodes & Physics
Interest
OpenGL
Interest
Overlay
Interest
Overrides
Interest
Performance
Interest
Physics
Interest
Pipeline, Assets & IO
Interest
Platforms, Builds & Tests
Interest
Python API
Interest
Render & Cycles
Interest
Render Pipeline
Interest
Sculpt, Paint & Texture
Interest
Text Editor
Interest
Translations
Interest
Triaging
Interest
Undo
Interest
USD
Interest
User Interface
Interest
UV Editing
Interest
VFX & Video
Interest
Video Sequencer
Interest
Virtual Reality
Interest
Vulkan
Interest
Wayland
Interest
Workbench
Interest: X11
Legacy
Blender 2.8 Project
Legacy
Milestone 1: Basic, Local Asset Browser
Legacy
OpenGL Error
Meta
Good First Issue
Meta
Papercut
Meta
Retrospective
Meta
Security
Module
Animation & Rigging
Module
Core
Module
Development Management
Module
EEVEE & Viewport
Module
Grease Pencil
Module
Modeling
Module
Nodes & Physics
Module
Pipeline, Assets & IO
Module
Platforms, Builds & Tests
Module
Python API
Module
Render & Cycles
Module
Sculpt, Paint & Texture
Module
Triaging
Module
User Interface
Module
VFX & Video
Platform
FreeBSD
Platform
Linux
Platform
macOS
Platform
Windows
Priority
High
Priority
Low
Priority
Normal
Priority
Unbreak Now!
Status
Archived
Status
Confirmed
Status
Duplicate
Status
Needs Info from Developers
Status
Needs Information from User
Status
Needs Triage
Status
Resolved
Type
Bug
Type
Design
Type
Known Issue
Type
Patch
Type
Report
Type
To Do
No Milestone
No project
No Assignees
5 Participants
Notifications
Due Date
The due date is invalid or out of range. Please use the format 'yyyy-mm-dd'.

No due date set.

Dependencies

No dependencies set.

Reference: blender/blender#86170
No description provided.