Skip to content
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

feat(repository): hasOne relation #2005

Merged
merged 3 commits into from
Dec 4, 2018
Merged

feat(repository): hasOne relation #2005

merged 3 commits into from
Dec 4, 2018

Conversation

b-admike
Copy link
Contributor

@b-admike b-admike commented Nov 9, 2018

**EDIT - Continuation of #1879 **

First iteration of hasOne relation. This PR includes the the repository factory, repository interface and default hasOne repository, and has one decorator implementations (mostly taken from our hasMany implementation since they're very similar in nature). It also includes an acceptance test with a Customer has one Address relation to drive it and a unit test as well. Please note that only the EDIT get and create methods are implemented at the moment and tests covering them, so that we can focus on that and incrementally add patch and delete methods to the hasOne repository interface.

Todos

  • Add documentation for hasOne relation
  • Add example that uses hasOne relation

See discussion in #1956. @bajtos proposed using the PK of a target model in a hasOne relation to enforce uniqueness and 1-to-1 relation between source and target model instances (see #1956 (comment)). This is just a quick spike to see the feasibility of the proposal and get more feedback.

Checklist

  • npm test passes on your machine
  • New tests added or existing tests modified to cover all changes
  • Code conforms with the style guide
  • API Documentation in code was updated
  • Documentation in /docs/site was updated
  • Affected artifact templates in packages/cli were updated
  • Affected example projects in examples/* were updated

@b-admike b-admike changed the base branch from master to feat/hasone-relation November 9, 2018 16:49
@b-admike b-admike changed the title [WIP Spike PoC]: Use target model PK for hasOne relation to enforce uniqueness [WIP PoC]: Use target model PK for hasOne relation to enforce uniqueness Nov 9, 2018
@bajtos
Copy link
Member

bajtos commented Nov 12, 2018

Looks good to me at high level 👍 But! One of the has-many tests is failing, so your implementation may need few tweaks?

I think the next task is to verify this implementation against real databases, e.g. MongoDB and MySQL. It would be great to have a full example app with hasOne relation configured for that. Can we come up with a meaningful entity that we could add to the existing todo-list example?

@b-admike b-admike mentioned this pull request Nov 21, 2018
20 tasks
@b-admike b-admike force-pushed the feat/hasone-relation branch from 2b6221b to fd03a78 Compare November 22, 2018 06:29
Copy link
Member

@bajtos bajtos left a comment

Choose a reason for hiding this comment

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

Uh oh, looks like I forgot to post few review comments the last time I was reviewing these changes.

@b-admike b-admike changed the base branch from feat/hasone-relation to master November 22, 2018 18:51
@b-admike b-admike force-pushed the spike/hasone-relation branch from ca49f91 to 3c96fea Compare November 22, 2018 18:59
@b-admike b-admike force-pushed the spike/hasone-relation branch from 3c96fea to 93384d9 Compare November 27, 2018 16:58
@b-admike b-admike mentioned this pull request Nov 29, 2018
7 tasks
@b-admike b-admike changed the title [WIP PoC]: Use target model PK for hasOne relation to enforce uniqueness feat(repository): hasOne relation Nov 29, 2018
examples/todo-list/src/models/author.model.ts Outdated Show resolved Hide resolved
examples/todo-list/src/controllers/author.controller.ts Outdated Show resolved Hide resolved
options,
);
if (found.length < 1) {
// We don't have a direct access to the foreign key value here :(
Copy link
Member

Choose a reason for hiding this comment

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

Should we leave a TODO comment here? Maybe open a new issue to follow-up later?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Opened #2114 as a follow-up issue.

Copy link
Member

@bajtos bajtos left a comment

Choose a reason for hiding this comment

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

Looks reasonably good for the first iteration.

Please address the comment containing exclamation mark ❗️ before landing this pull request.

Remaining comments can be addressed either as part of this PR or in a follow-up pull request.

@b-admike b-admike force-pushed the spike/hasone-relation branch 2 times, most recently from 3e3a9ac to 9dea156 Compare December 3, 2018 07:34
@@ -11,8 +11,11 @@ import {BelongsToDefinition, RelationType} from '../relation.types';

/**
* Decorator for belongsTo
* @param targetResolver
* @param definition
* @param targetResolver A resolver function that retruns the target model for
Copy link
Member

Choose a reason for hiding this comment

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

typo: retruns

@@ -132,7 +132,7 @@ describe('createHasOneRepositoryFactory', () => {
province: String;
}

class Order extends Entity {
class ModelWithoutIndexFK extends Entity {
static definition = new ModelDefinition('Order')
Copy link
Member

Choose a reason for hiding this comment

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

Please fix the model name string to match the class name.

@@ -142,18 +142,23 @@ describe('createHasOneRepositoryFactory', () => {
type: 'string',
required: true,
})
.addProperty('id', {
type: 'number',
id: true,
Copy link
Member

Choose a reason for hiding this comment

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

I am little bit confused. The model name is ModelWithoutIndexFK, but it has a regular id property. What am I missing?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Good question. I added an id property because the model itself extends from Entity, but the purpose here is the decorated relational property is not part of the composite index, thus the name ModelWithoutIndexFK. Thoughts on improving it if it's confusing? I can extend from Model and remove the id property as well.

Copy link
Member

Choose a reason for hiding this comment

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

I can extend from Model and remove the id property as well.

If this works, then I think it would be a good solution.

If not, then please capture the information from your comment above in a code comment.

Copy link
Contributor

@jannyHou jannyHou left a comment

Choose a reason for hiding this comment

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

@b-admike Great effort! I haven't got enough time to review each single detail but the new example and test cases LGTM in general 👍

I have a question regarding the BelongsToUnique decorator: My understanding is, regardless it's a hasMany relation or hasOne relation, one child instance could only have one parent, why would we create a specific belongsToUnique decorator for hasOne?

@b-admike b-admike force-pushed the spike/hasone-relation branch from a971555 to e128d86 Compare December 4, 2018 06:09
@b-admike
Copy link
Contributor Author

b-admike commented Dec 4, 2018

I have a question regarding the BelongsToUnique decorator: My understanding is, regardless it's a hasMany relation or hasOne relation, one child instance could only have one parent, why would we create a specific belongsToUnique decorator for hasOne?

Great question @jannyHou. The reason we created a sugar syntax decorator belongsToUniquely is so that the decorated property is designated as a PK for the target model (needed for hasOne relation). It is to save time to label it as an id property that is not generated. It makes the UX better to not manually write those out.

@b-admike b-admike force-pushed the spike/hasone-relation branch from e128d86 to b9c51b3 Compare December 4, 2018 06:20
@b-admike b-admike force-pushed the spike/hasone-relation branch from b9c51b3 to e57d9c1 Compare December 4, 2018 06:40
@b-admike b-admike merged commit 1b5b66a into master Dec 4, 2018
@b-admike b-admike deleted the spike/hasone-relation branch December 4, 2018 06:55
Copy link
Contributor

@raymondfeng raymondfeng left a comment

Choose a reason for hiding this comment

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

Please address my comments.

@property()
registeredOn: Date;
}

Copy link
Contributor

Choose a reason for hiding this comment

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

It's not so obvious that RegistrationDate deserves a hasOne relation from the Customer. Something like A customer has a public profile is probably better.

const propertyMetaDefaults: Partial<PropertyDefinition> = {
id: true,
generated: false,
};
Copy link
Contributor

Choose a reason for hiding this comment

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

Is the intention to make the FK unique? If so, I think it's wrong to set id to true as it will add the FK to the PK. For example, (id, customerId) becomes the PK.

I think we need to mark it as unique. IIRC, we can mark it uniquely indexed.

Copy link
Member

Choose a reason for hiding this comment

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

Is the intention to make the FK unique?

Yes.

If so, I think it's wrong to set id to true as it will add the FK to the PK. For example, (id, customerId) becomes the PK.

The way we have envisioned the current hasOne implementation, there is only one PK property, and that's customerId, there is no property called id. I.e. the primary key is a foreign key too.

I think we need to mark it as unique. IIRC, we can mark it uniquely indexed.

I am not sure if all databases support unique constraint. I am also pretty sure our auto-migration implementation does not correctly define unique constraint either.

As a result, by using unique instead of id, the default implementation of hasOne in LB4 would not work correctly out of the box.

On the other hand, all databases are enforcing the uniqueness constraint on primary keys (id property), otherwise primary keys could not be used as unique identifiers.

Copy link
Contributor

Choose a reason for hiding this comment

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

I think it's too restrictive to promote the FK for hasOne to be part of PK. At least we should check if there is an existing PK.

BTW, marking a property id: true does not guarantee it will be enforced by the database. I think we should generate an index with {unique: true} option per https://loopback.io/doc/en/lb3/Model-definition-JSON-file.html#indexes.

Copy link
Member

Choose a reason for hiding this comment

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

Not only we are promoting FK to be the only PK, but we also require the target od HasOne to have the same PK as the model it relates to. E.g. when a Customer has one PublicProfile, then both the customer instance and the profile instance share the same primary key value.

This is the only way I am aware of that offers strong guarantees about consistency and avoid possible race conditions in all databases we support in LoopBack.

I am more than happy to consider different approaches, as long as they work similarly well across the databases we have connectors for.

Should we open a new issue to continue this discussion?

* @param definition Optional metadata for setting up a belongsTo relation
* @returns {(target: Object, key:string)}
*/
export function belongsToUniquely<T extends Entity>(
Copy link
Contributor

Choose a reason for hiding this comment

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

The name is not easy to understand. The purpose is to say the FK should be unique.

Copy link
Contributor

Choose a reason for hiding this comment

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

Maybe it's a warning sign that we should not overload @belongTo for responsibility of @property.

Copy link
Member

Choose a reason for hiding this comment

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

I agree that ideally, relation decorators like @belongsTo and @hasMany should not call @property decorator under the hood. To get there, we would have to change the current design in a non-trivial way and break backwards compatibility along the way.

Do we want to delay hasOne implementation?

Personally, I'd prefer to finish hasOne implementation using the current approach, and then look into ways how to change the overall design.

The name is not easy to understand. The purpose is to say the FK should be unique.

Can you propose a better name that's easier to understand?

Our intention here is to provide a syntactic sugar that will allow users to tell what they want to accomplish (setup the other side of hasOne relation) without going into low-level details (setting up a PK property as a way to get uniqueness constraint + make queries fast).

@RaphaelDrai
Copy link

Hello @b-admike ,
Great work with the merge of hasOne, thanks you so much !
As discussed in #1879 we are currently working on hasOne example and documentation as per todos requirement above.
Is there an estimation when the missing PATCH and DELETE operations will be covered? If not, and that no one is actually working on PATCH and DELETE, we can contribute to provide these features. Let me know if we can start this task?
Raphael

@b-admike
Copy link
Contributor Author

b-admike commented Dec 6, 2018

hey @RaphaelDrai, sorry just saw your comment; based on the comments from @raymondfeng and @bajtos and discussion with @raymondfeng, I've decided to try the unique index approach for the FK relation property on the target model for a hasOne relation. With it, I plan to add the PATCH and DELETE operations and docs for the hasOne feature. Are you able to open a PR with your work on the example and its documentation? If so, we can definitely help you land it 👍

@RaphaelDrai
Copy link

Hello @b-admike,
thanks you for the update about the DELETE and PATCH operations, this is great... 🥇
Just for our internal knowledge, do you have a time estimation when they will be available?
Yes, we will be very happy to get your help of how creating a PR with our work and how to publish our hasOne example and documentation.

@b-admike
Copy link
Contributor Author

Hello @b-admike,
thanks you for the update about the DELETE and PATCH operations, this is great... 🥇
Just for our internal knowledge, do you have a time estimation when they will be available?
Yes, we will be very happy to get your help of how creating a PR with our work and how to publish our hasOne example and documentation.

@RaphaelDrai Right now I'm using #2126 for enforcement of uniqueness on the target model instead of marking the foreign key as a primary key in the current design. @bajtos @raymondfeng and I had a discussion and since the memory connector and cloudant connector do not support unique indexes, we've decided to have the first release of hasOne provide "weak" relations i.e. ease of navigation and aggregation instead of enforcement of referential integrity and document it so users are aware of the limitations. We will set the metadata for the foreign key to be unique index, but it is up to users to run automigrate and if not supported, to set up the referential integrity in their collections/tables. I'm working on DELETE and PATCH operations at the moment in another branch as well as supporting unique indexes for memory connector (see loopbackio/loopback-datasource-juggler#1672) and I would like to ideally wrap them up this week. Can you let us know a bit more about your use case for the hasOne relation? I.e. what are your tables defined as and what is your expectation to enforce unique constraint?

@RaphaelDrai
Copy link

Hello @b-admike,
thanks you for the update about the DELETE and PATCH operations, this is great... 🥇
Just for our internal knowledge, do you have a time estimation when they will be available?
Yes, we will be very happy to get your help of how creating a PR with our work and how to publish our hasOne example and documentation.

@RaphaelDrai Right now I'm using #2126 for enforcement of uniqueness on the target model instead of marking the foreign key as a primary key in the current design. @bajtos @raymondfeng and I had a discussion and since the memory connector and cloudant connector do not support unique indexes, we've decided to have the first release of hasOne provide "weak" relations i.e. ease of navigation and aggregation instead of enforcement of referential integrity and document it so users are aware of the limitations. We will set the metadata for the foreign key to be unique index, but it is up to users to run automigrate and if not supported, to set up the referential integrity in their collections/tables. I'm working on DELETE and PATCH operations at the moment in another branch as well as supporting unique indexes for memory connector (see strongloop/loopback-datasource-juggler#1672) and I would like to ideally wrap them up this week. Can you let us know a bit more about your use case for the hasOne relation? I.e. what are your tables defined as and what is your expectation to enforce unique constraint?

Thanks you for your detailed and helpful answer.
Our particular use case for the 'hasOne' relation is creating a connection from a model let's say "Country" with another model let's say "Capital City". The relation should follow the "one-to-one" relation where a given "Country" can have only one "Capital City" and vice versa, a given "Capital City" can have (belongs to) only one "Country".

I read the previous messages about "belongsToUnique" decorator. I think that it could be helpful to have the option to use "belongsTo" or "belongsToUnique" in 'hasOne' relation.
For example if we use the decorator "belongsToUnique" it will address our use case above where in both directions the one-to-one relation is preserved and ensured.
But if we use the decorator "belongsTo" it will address the use case of one-to-one in one direction, and in the opposite direction we can allow one-to-many.
Example of this use case is a data model "Customer" connected to a data model "Orders" using the 'hasOne' relation. A given "Order" can have only one Customer, this is enforced with the 'belongsToUnique', but in the opposite direction, "Customer" can have many orders.

@b-admike
Copy link
Contributor Author

Thanks you for your detailed and helpful answer.
Our particular use case for the 'hasOne' relation is creating a connection from a model let's say "Country" with another model let's say "Capital City". The relation should follow the "one-to-one" relation where a given "Country" can have only one "Capital City" and vice versa, a given "Capital City" can have (belongs to) only one "Country".

I read the previous messages about "belongsToUnique" decorator. I think that it could be helpful to have the option to use "belongsTo" or "belongsToUnique" in 'hasOne' relation.
For example if we use the decorator "belongsToUnique" it will address our use case above where in both directions the one-to-one relation is preserved and ensured.
But if we use the decorator "belongsTo" it will address the use case of one-to-one in one direction, and in the opposite direction we can allow one-to-many.
Example of this use case is a data model "Customer" connected to a data model "Orders" using the 'hasOne' relation. A given "Order" can have only one Customer, this is enforced with the 'belongsToUnique', but in the opposite direction, "Customer" can have many orders.

Thanks for your feedback @RaphaelDrai. Those sound like good use cases. Unfortunately, I'm going to revert belongsToUnique implementation since it marks the foreign key as a primary key on the target model (in this case Capital City). We have agreed to let our users/DBAs set up the unique constraint(s) for the relation and provide the "weak" relation I've mentioned above. I have a PR on the way and I will ping you as well to give your feedback/review.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants