-
Notifications
You must be signed in to change notification settings - Fork 349
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 for Composite Keys using @Embeded
and @Id
#574
Comments
Lovro Pandžić commented Hello, Jens Schauder, any idea when this might get fixed? Composite keys seem to be a pretty fundamental feature for any persistence layer library. Since there are no workarounds (I don't consider adding a new unique key to a production table because of library limitations a viable workaround), users that do use composite keys can't use Spring Data JDBC |
Jens Schauder commented This is a feature we want to have. Unfortunately we currently can't see far enough in the future to tell when it will happen. |
hi |
1 similar comment
hi |
meanwhile this feature unsupported yet is there a workaround for this? |
Hi |
I managed to make it work, with a workaround. Working example in Kotlin (it would work also in a Java equivalent): // Database table model
@Table("policy")
class Policy(
val userId: UUID,
val entityId: UUID,
val name: String,
)
// Composite primary key
data class PolicyPK(
val userId: UUID? = null,
val entityId: UUID? = null,
) : Serializable
// Repository
interface PolicyRepository : R2dbcRepository<Policy, PolicyPK> {
/**
* replaces: findById(id: PolicyPk)
* Find by id does not work because of the composite id so we query with all id fields
*/
fun findByUserIdAndEntityId(userId: UUID, entityId: UUID): Mono<Policy>
/**
* replaces: delete(policy)
* Delete with entity does not work because of the composite id so we delete with all id fields
*/
fun deleteByUserIdAndEntityId(userId: UUID, entityId: UUID): Mono<Void>
}
If you really need to have the default I hope this helps someone, spend a bunch of time to figure this out |
Just in case if somebody wants to know @porfirioribeiro's workaround (no id property) does not work with R2dbcEntityTemplate |
@porfirioribeiro your way also can't save any record, the application throws me "IllegalStateException: Required identifier property not found " |
@xuandungdoan you're right @Modifying
@Query(
"""
insert into policy.policy(user_id, entity_id, resource_id, resource_type, roles, created_at, updated_at)
values (:#{#policy.userId}, :#{#policy.entityId}, :#{#policy.resourceId}, :#{#policy.resourceType}, :#{#policy.roles}, :#{#policy.createdAt}, now())
on conflict (user_id, entity_id, resource_id) do update set roles=:#{#policy.roles}, updated_at=now()"""
)
fun upsert(policy: Policy): Mono<Int> It kinda sucked to have to do all this workaround to make it work, but at least I could still be using R2DBC and part of the Repository stuff, and not have to change all this back to use JPA/Hibernate |
Any updates on this? I need this feature to be able refer an aggregate across multiple roots using a pojo class representing the composite key |
* [#28] security 적용 (SecurityConfig.kt) , api mock 생성 * [#28] 가입 인증코드 기능구현 * [#28] 가입기능 구현 * [#28] authCode 복합키로 설정 - 복합키 미지원 확인 spring-projects/spring-data-relational#574 * [#28] spring security에서 swagger관련 제외 * [#28] 이벤트 통일 , exception 응답 설정, 트랜잭션 커밋안되는 이슈 수정 * [#28] 이전 상태 체크 추가 * Update user-api/src/main/kotlin/com/sns/user/core/config/SwaggerTag.kt Co-authored-by: Chanhyeong Cho <[email protected]> * PR반영, 논의내용 반영 - response는 json으로 - crudRepository 활용 - controller에 함께 aggregator 위치. (1:1 매칭) * 코드정리, test fail 수정 Co-authored-by: Chanhyeong Cho <[email protected]>
Oh dear, this is a bit of an issue. I am trying to move a legacy C# application into Spring and just came across this issue when trying to use Spring JDBC.
One thing I have noticed is that using these two together may not work that well for other autogenerated methods - Perhaps having multiple I may be able to contribute a PR for this, can anyone from the Spring JDBC team give me any guidance on approach? Would |
We have issues with this every day. I am slowly coming to terms with the fact that this is not coming any time soon. This clashes so much with sensible DDD decisions we made. Our company will monitor if there is any movement regarding this, and I'm pretty sure we are stepping away from Spring Data Relational, as promising as it was, because this breaks so much stuff. |
This is almost on the top of the priority list. Right after the current stuff I'm working which unfortunately is beyond big. |
Good to know @schauder ! Is there a roadmap somewhere that's publicly visible? Thanks 😊👍 |
That is great news in itself. Thank you very much for clearing that up! In that case, we're holding out a little longer. Under different circumstances I personally would have also been interested in contributing, but unfortunately my schedule doesn't allow that this year. |
Is there a roadmap and plan to fix this any sooner? |
This is absolutely required. 🙏 |
@Embeded
and @Id
Support for composite keys in a database table is essential. Otherwise, this software just isn't suitable for enterprise database solutions. Forcing a generated unique surrogate key for every single database table is simply unacceptable in the real world. Take the example of a many-many relationship resolution table with PKs from other tables. Why do I need an additional generated key just for this? It's an utter waste of valuable resources. Composite keys aren't just in use in "legacy" databases, they have perfectly valid uses in any physical database design depending on the business and performance requirements. Please prioritise this or your software is not fit for purpose in the real world of physical database design. |
@morfsnz Spring Data JDBC does not require an additional generated (or otherwise) key for m-to-n relationships. |
Agree on that BUT as of the topic of this conversation is related to the fact that R2DBC does not support basic things such as composite keys for instance. If you compare with other implementations of reactive connection, such as the one used on quarkus, there are things there that are supported through the use of Hibernate for example. Anyway, any timeline on when this will be worked on? |
This is a nice pun as a basic thing is a basic key. Anyway, we do not have a timeline except for "at some point in the future" as the team is currently busy with higher-priority topics. We always welcome contributions that help us and the project to get things done and to speed things up. |
@morfsnz Would be interesting if you could provide som real world examples / use-cases so I could explore possible solutions using Spring Data JDBC. In general, I have tried to give my 2 cents on possible handling this on StackOverflow. |
Hey everyone! Just want to add a quick note. As it was written above, @porfirioribeiro's workaround did not work for Unfortunately, it still doesn't work properly, and I could only write a custom UPSERT query. But anyway, thanks @porfirioribeiro for his workaround. And thanks to Spring developers for their product, I hope this feature will be implemented in the near future :) |
Any info on when this basic feature will be taken into consideration? |
It's been 5 years and this still has not been addressed! Why? It's such a core feature of any ORM. |
this will be such a nice feature to have! |
Any news on this topic? @schauder, I would like to add that in version 3.2.5 I tested composite keys using
Maybe it was a coincidence but it could be a start for the development of the feature. |
With the recently released version 3.3.3 including changes from #1502 the last workaround for us - related to multiple columns as primary key - stopped to work. We used something like the following to work with a transformation table of an many-to-many relation where two columns in combination are the primary key. CREATE TABLE authors_books (
author_id BIGINT,
book_id BIGINT,
PRIMARY KEY (author_id, book_id)
); @Table("authors_books")
data class AuthorBookRelationEntity(
@Column("author_id")
val authorId: Long,
@Column("book_id")
val bookId: Long,
) : Persistable<AuthorBookRelationEntity> {
@Transient
var isNewInstance: Boolean? = false
override fun getId(): AuthorBookRelationEntity {
return this
}
override fun isNew(): Boolean {
return isNewInstance ?: false
}
} With v3.3.2 it worked fine, but with v3.3.3 we get the exception Nevertheless, it would be awesome to have the possibility to use multiple columns as identifier. |
Non trivial aggregates work with single value wrapped PK. Simple aggregates work with composite id for insert, update, delete and exists. See #574
Non trivial aggregates work with single value wrapped PK. Simple aggregates work with composite id for insert, update, delete and exists. See #574
Any news on this one? I'd really like to switch to this from JPA. |
It is now possible to use composite ids, by having an entity reference annotated with `@Id` and `@Embedded`. Closes #574
Entities may be annotated with `@Id` and `@Embedded`, resulting in a composite id on the database side. The full embedded entity is considered the id, and therefore the check for determining if an aggregate is considered a new aggregate requiring an insert or an existing one, asking for an update is based on that entity, not its elements. Most use cases will require a custom `BeforeConvertCallback` to set the id for new aggregate. For an entity with `@Embedded` id, the back reference used in tables for referenced entities consists of multiple columns, each named by a concatenation of <table-name> + `_` + <column-name>. E.g. the back reference to a `Person` entity, with a composite id with the properties `firstName` and `lastName` will consist of the two columns `PERSON_FIRST_NAME` and `PERSON_LAST_NAME`. This holds for directly referenced entities as well as `List`, `Set` and `Map`. Closes #574
Entities may be annotated with `@Id` and `@Embedded`, resulting in a composite id on the database side. The full embedded entity is considered the id, and therefore the check for determining if an aggregate is considered a new aggregate requiring an insert or an existing one, asking for an update is based on that entity, not its elements. Most use cases will require a custom `BeforeConvertCallback` to set the id for new aggregate. For an entity with `@Embedded` id, the back reference used in tables for referenced entities consists of multiple columns, each named by a concatenation of <table-name> + `_` + <column-name>. E.g. the back reference to a `Person` entity, with a composite id with the properties `firstName` and `lastName` will consist of the two columns `PERSON_FIRST_NAME` and `PERSON_LAST_NAME`. This holds for directly referenced entities as well as `List`, `Set` and `Map`. Closes #574
alpet opened DATAJDBC-352 and commented
Enable usage of
@Embdedded
and@Id
together. I see two scenarios here. One would be for read only repositories and the other for the CRUD repository. With respect of the legacy applications very often read only is enough. How would Embedded participate in a query ?One reason to use a lower level persistence framework in my opinion are legacy applications who very often have Composite Keys. Emabling such feature in spring data jdbc would increase the possible users of the framework
13 votes, 14 watchers
The text was updated successfully, but these errors were encountered: