No edit summary |
Hans Karlsen (talk | contribs) No edit summary |
||
(6 intermediate revisions by 2 users not shown) | |||
Line 1: | Line 1: | ||
==== Declaring a Composite key for a reversed database ==== | |||
In this video: https://youtu.be/BmmoAFjaQtc the process of defining a composite primary key is explained. | |||
Here are the steps involved: | |||
# On the class (filmactor) find the two foreign keys that together makes the row unique in the database (film_id,actor_id) | |||
# Set the both these keys as comma separated on the class (A) (filmactor) primarykey: film_id,actor_id | |||
# In the singlelink ends (film (B) and actor (C)) set the columnName (D) to film_id and actor_id respectively | |||
[[File:Training Custom OR Mapping 1708350261846.png|none|thumb|674x674px]] | |||
==== Setting the primary key to an existing attribute ==== | |||
We had this issue from Rich W in the forum – he tried to do a custom OR mapping. | We had this issue from Rich W in the forum – he tried to do a custom OR mapping. | ||
Line 7: | Line 19: | ||
[[File:Custom OR Mapping 01.png|frameless|249x249px]] | [[File:Custom OR Mapping 01.png|frameless|249x249px]] | ||
In Class1 we | In Class1, we make SomePK the primary key. | ||
On Class1 we set PrimaryKey and PrimaryKeyMapper: | On Class1, we set PrimaryKey and PrimaryKeyMapper: | ||
[[File:Custom OR Mappin 02.png|frameless]] | [[File:Custom OR Mappin 02.png|frameless]] | ||
Line 17: | Line 29: | ||
[[File:Custom OR Mappin 03.png|frameless]] | [[File:Custom OR Mappin 03.png|frameless]] | ||
Doing this | Doing this means that we take responsibility for setting the key and we do not allow it to be changed once saved – as is normal for primary keys. | ||
If we had DBAssigned – it will not be set on save – and it will be re-read from the DB once saved. This is for scenarios where you have AutoInc fields in the database. | If we had DBAssigned – it will not be set on save – and it will be re-read from the DB once saved. This is for scenarios where you have AutoInc fields in the database. | ||
Rich did all this correctly but could not get it to work – he got errors while trying to generate the database. At first I could not put my finger on why but then it dawned on me that we have an issue with the DefaultSuperClass. | Rich did all this correctly but could not get it to work – he got errors while trying to generate the database. At first, I could not put my finger on why but then it dawned on me that we have an issue with the DefaultSuperClass. | ||
The package may have a named DefaultSuperClass that all the other classes in the package inherit from. | The package may have a named DefaultSuperClass that all the other classes in the package inherit from. This still has the default primary key mapping. | ||
If you do not have a designed DefaultSuperClass | If you do not have a designed DefaultSuperClass, the auto-generated class EcoModelRoot will assume its place in the standard OR mapping. | ||
So the EcoModelRoot has a standard key named EcoId and the subclass Class1 has the SomePK attribute as primary key – | So the EcoModelRoot has a standard key named EcoId and the subclass Class1 has the SomePK attribute as the primary key – this is both strange and not what Rich wanted – and this is also unsupported – hence the error Rich got. | ||
To handle this we | To handle this, we instruct the OR mapper to do something about it. In this particular case, the best thing is to skip the EcoModelRoot table. We do this by adding a DefaultORMappingBuilder to the PersistenceMapper: | ||
[[File:Custom OR Mappin 04.png|frameless|400x400px]] | [[File:Custom OR Mappin 04.png|frameless|400x400px]] | ||
We connect it to both NewMappingProvider and RunTimeMappingProvider of the PersistenceMapper (New is used in design time when doing the create and evolve | We connect it to both NewMappingProvider and RunTimeMappingProvider of the PersistenceMapper (New is used in design time when doing the create and evolve and runtime is when the app executes): | ||
[[File:Custom OR Mapping 05.png|frameless|388x388px]] | [[File:Custom OR Mapping 05.png|frameless|388x388px]] | ||
So far we have | So far, we have changed nothing – this is what would have happened anyway – but we have exposed the defaultORMappingBuilder so that we can tweak it: | ||
[[File:Custom OR Mapping 06.png|frameless|384x384px]] | [[File:Custom OR Mapping 06.png|frameless|384x384px]] | ||
The most important tweak is ChildMapRootClass – this means that EcoModelRoot | The most important tweak is ChildMapRootClass – this means that EcoModelRoot, or if you have another modeled DefaultSuperClass, will not get a table of its own – it will be mapped into the child classes. The other tweaks are just for show – I do this because I personally think it looks better with primary keys that follow the pattern above (original value is <Name> and EcoKey). | ||
Having this in place, I can generate a schema and get the following result: | |||
[[File:Custom OR Mapping 07.png|frameless]] | [[File:Custom OR Mapping 07.png|frameless]] | ||
Class1 has the SomePK | Class1 has the SomePK as the key – but Class2, which we did not give any special primary key, has the default <TableName>ID – namely Class2ID. Also, note the single link back to Class1 in the Class2 table – it is named Class1ID. | ||
[[Category:Advanced]] | [[Category:Advanced]] | ||
[[Category:Database]] |
Latest revision as of 13:45, 19 February 2024
Declaring a Composite key for a reversed database
In this video: https://youtu.be/BmmoAFjaQtc the process of defining a composite primary key is explained.
Here are the steps involved:
- On the class (filmactor) find the two foreign keys that together makes the row unique in the database (film_id,actor_id)
- Set the both these keys as comma separated on the class (A) (filmactor) primarykey: film_id,actor_id
- In the singlelink ends (film (B) and actor (C)) set the columnName (D) to film_id and actor_id respectively
Setting the primary key to an existing attribute
We had this issue from Rich W in the forum – he tried to do a custom OR mapping.
He wanted to pick a modeled attribute to be the primary key in the database.
This is how to accomplish this:
In Class1, we make SomePK the primary key.
On Class1, we set PrimaryKey and PrimaryKeyMapper:
We also set the SaveAction on the attribute SomePK to Freeze – possible other values DBAssigned and None.
Doing this means that we take responsibility for setting the key and we do not allow it to be changed once saved – as is normal for primary keys.
If we had DBAssigned – it will not be set on save – and it will be re-read from the DB once saved. This is for scenarios where you have AutoInc fields in the database.
Rich did all this correctly but could not get it to work – he got errors while trying to generate the database. At first, I could not put my finger on why but then it dawned on me that we have an issue with the DefaultSuperClass.
The package may have a named DefaultSuperClass that all the other classes in the package inherit from. This still has the default primary key mapping.
If you do not have a designed DefaultSuperClass, the auto-generated class EcoModelRoot will assume its place in the standard OR mapping.
So the EcoModelRoot has a standard key named EcoId and the subclass Class1 has the SomePK attribute as the primary key – this is both strange and not what Rich wanted – and this is also unsupported – hence the error Rich got.
To handle this, we instruct the OR mapper to do something about it. In this particular case, the best thing is to skip the EcoModelRoot table. We do this by adding a DefaultORMappingBuilder to the PersistenceMapper:
We connect it to both NewMappingProvider and RunTimeMappingProvider of the PersistenceMapper (New is used in design time when doing the create and evolve and runtime is when the app executes):
So far, we have changed nothing – this is what would have happened anyway – but we have exposed the defaultORMappingBuilder so that we can tweak it:
The most important tweak is ChildMapRootClass – this means that EcoModelRoot, or if you have another modeled DefaultSuperClass, will not get a table of its own – it will be mapped into the child classes. The other tweaks are just for show – I do this because I personally think it looks better with primary keys that follow the pattern above (original value is <Name> and EcoKey).
Having this in place, I can generate a schema and get the following result:
Class1 has the SomePK as the key – but Class2, which we did not give any special primary key, has the default <TableName>ID – namely Class2ID. Also, note the single link back to Class1 in the Class2 table – it is named Class1ID.