Skip to content

Conversation

@mipo256
Copy link
Contributor

@mipo256 mipo256 commented Jul 26, 2025

Closes #2099
Closes #1986

This PR enhances the RelationalExampleMapper by making it consider embedded properties and specifiers for them.

As it turned out during resolution of #1986, the embedded properties were kind of ignored in QBE.

Current algorithm works more a less the same way, it just traverses the Example recursively from direct properties defined in the root probe, all the way down to the deeply embedded leafs.

Tests are included in the PR.

CC: @mp911de @schauder

@spring-projects-issues spring-projects-issues added the status: waiting-for-triage An issue we've not yet triaged label Jul 26, 2025
@mp911de mp911de added type: enhancement A general enhancement and removed status: waiting-for-triage An issue we've not yet triaged labels Aug 4, 2025
@mipo256
Copy link
Contributor Author

mipo256 commented Aug 5, 2025

Signed-off the commits

@mipo256 mipo256 force-pushed the GH-2099 branch 3 times, most recently from 3d8fa6e to a7f08ef Compare August 5, 2025 20:40
Copy link
Member

@mp911de mp911de left a comment

Choose a reason for hiding this comment

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

This PR goes into the right direction. There's quite some noise and parameter and variable names are quite lengthy. Long names add complexity and from my perspective, they do not improve readability or understandability by explaining different domains.


Object actualPropertyValue = entityPropertiesAccessor.getProperty(property);

if (property.isEmbedded() && actualPropertyValue != null) {
Copy link
Member

Choose a reason for hiding this comment

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

Ideally, we don't distinguish based on whether the property is an embedded one but rather whether it is an entity. At some point it should be possible to express criteria based across objects that are part of the aggregate (like it is done in JPA, we consider embeddables and references).

* @return query
*/
private <T> Query getMappedExample(Example<T> example, RelationalPersistentEntity<?> entity) {
private <T> Query getMappedExample(Example<T> example, RelationalPersistentEntity<?> persistentEntity) {
Copy link
Member

Choose a reason for hiding this comment

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

What's the reason behind the renaming?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Because one can confuse the PersistenceEntity with the entity as the actual instance/probe. I do not insist here at all, just my observation when working with code, what do you think, @mp911de ?

Copy link
Contributor

Choose a reason for hiding this comment

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

I actually do like the rename and the idea behind it.

@mipo256
Copy link
Contributor Author

mipo256 commented Aug 16, 2025

@mp911de Just to make sure that we're on the same page. The current version of the algorithm accounts for the deeply nested embedded objects, like:

@Table
class Root {

    @Id
    private Long id;

    private String name;

    @Embedded.Nullable
    private EmbbedLevelOne levelone;
    
    public static class EmbeddedLevelOne { 
      
         private String type;
    
         @Embedded.Nullable
         private EmbeddedLevelTwo levelTwo;
    }
}

The problem is that if we unroll the recursion into the for-loop, then the code may get a bit messy, since this is not a tail-recursive function we're dealing with. I've tinkered with it, and the only sensible approach I found is to flatten-out all the properties of all the embedded objects (included deeply embedded) on demand, and store them in some kind of Queue that we can iterate through and then just perform the checking.

I'm unsure that this is the best solution. From my personal, empirical data, I may humbly conclude that objects. Therefore the call stack will probably not be large at all. But again, this is merely my observation.

Now, having said that, I can push the for-loop like approach with Queue and we'll have a look. WDYT, @mp911de, @schauder ?

@mipo256 mipo256 requested a review from mp911de August 16, 2025 11:58
@schauder
Copy link
Contributor

The problem is that if we unroll the recursion into the for-loop, then the code may get a bit messy, since this is not a tail-recursive function we're dealing with.

I think you are beyond what @mp911de asked for. His ask was just replacing the

persistentEntity.doWithProperties((PropertyHandler<RelationalPersistentProperty>) property -> {
   // ...
}

With a for loop:

for (RelationalPersisntentProperty property : persistentEntity) {
    // ....
} 

Not an unrolling of the recursion.

The doWithProperties creates one extra frame on the call stack.

@schauder schauder added the status: waiting-for-feedback We need additional information before we can continue label Sep 1, 2025
@dschulten
Copy link

@mipo256 can you change it to the suggested for loop?

@mipo256
Copy link
Contributor Author

mipo256 commented Oct 12, 2025

Sure @dschulten , I have already prepared necessary changes that address the review comments. I'll ping Mark and Jens once they are ready :)

@spring-projects-issues spring-projects-issues added status: feedback-provided Feedback has been provided and removed status: waiting-for-feedback We need additional information before we can continue labels Oct 12, 2025
mipo256 and others added 2 commits October 12, 2025 22:00
…rties in QBE

Signed-off-by: mipo256 <mikhailpolivakha@gmail.com>
Signed-off-by: Михаил Поливаха <mikhail.polivakha@gazprombank.ru>
Signed-off-by: Mikhail Polivakha <mikhailpolivakha@email.com>
Singed-off-by: mipo256 <mikhailpolivakha@email.com>
@mipo256
Copy link
Contributor Author

mipo256 commented Oct 12, 2025

Hey @mp911de, @schauder. I think, the PR is ready for the second round of review

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

Labels

status: feedback-provided Feedback has been provided type: enhancement A general enhancement

Projects

None yet

Development

Successfully merging this pull request may close these issues.

Embedded objects are mostly ignored in QBE queries JDBC: QBE withIgnoreNullValues does not ignore nulls in embedded classes of the Example

5 participants