-
-
Notifications
You must be signed in to change notification settings - Fork 263
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
Support nested DTO mapping #2540
Comments
Use I think that is what you are looking for. |
I saw this flag, but for nullable fields it becomes indistinguishable when this property is unloaded and when this property is null. |
Only if you don't know if the property was part of the projection AND you don't care to use Ebean's API / BeanState to determine if the property was loaded.
You need to explain more fully what the use case is, what you are trying to achieve and why |
Sure! For read use cases we use the following steps:
But it is hard to verify, that projection classes contains only selected fields.
I don't really like both options. I want at least to prohibit lazy load for this query and fail fast behavior which allows me to detect such problems in my tests relatively fast. But ideal solution for me looks like this:
Performance of this solution can be much better than select entities, because projections are read only and don't need any additional state. Other sql/orm frameworks have similar functionality: |
So one problem/issue is that we want to have a failing fast mechanism when doing the partial query + mapping it to DTO's + when the mapping is incorrect (e.g. the DTO property name and entity property names don't match and this isn't in the mapping). For this case we want to detect this and throw an exception rather than lazy load or leave null. Something like
While this is absolutely true of Hibernate generally this is not the case with Ebean. More specifically with Hibernate (using dynamic proxies) - 1. Hibernate doesn't really support partially loaded entities and 2. For dirty detection it needs to store the values additionally in the session - this means it does use a lot more memory regardless of whether the entity is read only or not. So yes, for Hibernate DTO queries are considered a lot cheaper than entity queries for those reasons and this approach is promoted a LOT (and so then a lot of people generally assume Ebean has similar issues which it doesn't). This isn't the case for Ebean because we do dirty detection on the beans themselves and this isn't any extra cost until you mutate the bean itself. That is, there is no extra state or memory consumption or cost with Ebean. For Ebean there is very little difference between a DTO query and an entity query in terms of cost. Hibernate with enhancement similarly reduces their problem but they still don't support partial objects yet so still push DTO queries. If we projected straight to a "nested DTO Graph" with Ebean we'd approximately be saving extra instantiation of entity beans + the mapping code execution (mapping entity -> dto).
Blaze here is mapping to interfaces and not to DTO's per say so these are Blaze specific "proxies". As I see it we would prefer plain old DTO's that have no attachment to a framework/library. For myself, I have always seen EntityView as addressing specific limitations with Hibernate that Ebean does not have rather than addressing the issue of "projecting to a complex DTO graph". Do I have the wrong impression about this?
You will have noticed that we literally don't need that annotation with Ebean. We can use constructors in our DTO's without needing anything explicit like that. In general the issue wrt nested DTO mapping is that "nested" translates to cardinality *ToOne or *ToMany ... and we are now building a graph. If we want to build a "consistent" graph then we need to know "identity" and use a "persistence context" to de-duplicate instances etc. So we pretty quickly either need to have DTO's that start looking like entity beans (but ideally we don't annotate the DTO's at all) ... OR ... derive some meta data (property mapping) that maps Id properties and something like simulate them being "read only entities". Hmmm. |
Thank you very much for so fast, complete and helful responses! You got first part absolutely correct, QueryDSL can map to plain dtos with dynamic constructor invocation like ebean. But i think that
So i think map to a plain dto is good, but if some annotation can integrate my dto with type safe query builder I will prefer this approach. You are absolutely right about mapping to nested dtos, I think them will be look like read only entities. I think, the main problem of partial objects is not technical (i don't care about several null fields) but is type safety. Currently I'm trying to use theese approches with ebean:
All of them technically solve my problem, but have some limitations and drawbacks. |
Just to note that the QueryDSL query could be written in Ebean with type safe projection and predicates like: List <UserInfo> result = new QUser()
.valid.eq(true)
.select(firstName, lastName)
.asDto(UserInfo.class) // turn it into a DtoQuery last
.findList(); |
Yes, but the main difference is that
|
Just to say that I have had a couple of thoughts on this and probably had an idea to try out with a view of supporting mapping into "nested dto's that contain ToOne and ToMany style relationships" (as opposed to "flat only dtos"). @belovaf If you put up an example repo of what QueryDSL was doing I'd take a look. I'd be especially interested in an example that "wasn't flat" / where the DTO's contained ToMany or ToOne or both. So an update to say there is something here I'm keen to try out and hopefully I get a crack at that in the next month or so. |
The first part of the plan for this is PR #2626 - going into Ebean 13.6.0. This provides an "intercept object" optimised for the readOnly + disableLazyLoading case - that is, the case where we build an ORM graph and want to map that into a plain DTO graph (of arbitrary complexity, ie. include ToMany / ToOne etc). |
Ebean perfectly supports fetching partial entities.
But it feels very error prone to return a partially loaded object for further processing.
Someone can accidentally access unloaded field and trigger lazy loading query.
I think it is a good idea to introduce a dto in such cases to prevent additional queries.
If someone wants to access an unloaded field he will be forced to update query and dto.
But currently ebean supports only plain dtos. This is not convenient to construct nested projections from a flat resultset manually.
It is possible to define new entity with
@View
annotation and map it to the same table, but it is not feels right.May be a better option exists, can you explain, please, how do you solve such problems?
The text was updated successfully, but these errors were encountered: