CQRS Event Sourcing: Validate UserName uniqueness

前端 未结 8 1256
醉话见心
醉话见心 2021-01-29 18:08

Let\'s take a simple \"Account Registration\" example, here is the flow:

  • User visit website
  • Click \"Register\" button and fill form, click \"Save\" button
相关标签:
8条回答
  • 2021-01-29 18:38

    About uniqueness, I implemented the following:

    • A first command like "StartUserRegistration". UserAggregate would be created no matter if user is unique or not, but with a status of RegistrationRequested.

    • On "UserRegistrationStarted" an asynchronous message would be sent to a stateless service "UsernamesRegistry". would be something like "RegisterName".

    • Service would try to update (no queries, "tell don't ask") table which would include a unique constraint.

    • If successful, service would reply with another message (asynchronously), with a sort of authorization "UsernameRegistration", stating that username was successfully registered. You can include some requestId to keep track in case of concurrent competence (unlikely).

    • The issuer of the above message has now an authorization that the name was registered by itself so now can safely mark the UserRegistration aggregate as successful. Otherwise, mark as discarded.

    Wrapping up:

    • This approach involves no queries.

    • User registration would be always created with no validation.

    • Process for confirmation would involve two asynchronous messages and one db insertion. The table is not part of a read model, but of a service.

    • Finally, one asynchronous command to confirm that User is valid.

    • At this point, a denormaliser could react to a UserRegistrationConfirmed event and create a read model for the user.

    0 讨论(0)
  • 2021-01-29 18:40

    I think for such cases, we can use a mechanism like "advisory lock with expiration".

    Sample execution:

    • Check username exists or not in eventually consistent read model
    • If not exists; by using a redis-couchbase like keyvalue storage or cache; try to push the username as key field with some expiration.
    • If successful; then raise userRegisteredEvent.
    • If either username exists in read model or cache storage, inform visitor that username has taken.

    Even you can use an sql database; insert username as a primary key of some lock table; and then a scheduled job can handle expirations.

    0 讨论(0)
  • 2021-01-29 18:40

    Like many others when implementing a event sourced based system we encountered the uniqueness problem.

    At first I was a supporter of letting the client access the query side before sending a command in order to find out if a username is unique or not. But then I came to see that having a back-end that has zero validation on uniqueness is a bad idea. Why enforce anything at all when it's possible to post a command that would corrupt the system ? A back-end should validate all it's input else you're open for inconsistent data.

    What we did was create an index table at the command side. For example, in the simple case of a username that needs to be unique, just create a user_name_index table containing the field(s) that need to be unique. Now the command side is able to query a username's uniqueness. After the command has been executed it's safe to store the new username in the index.

    Something like that could also work for the Order discount problem.

    The benefits are that your command back-end properly validates all input so no inconsistent data could be stored.

    A downside might be that you need an extra query for each uniqueness constraint and you are enforcing extra complexity.

    0 讨论(0)
  • 2021-01-29 18:48

    If you validate the username using the read model before you send the command, we are talking about a race condition window of a couple of hundred milliseconds where a real race condition can happen, which in my system is not handled. It is just too unlikely to happen compared to the cost of dealing with it.

    However, if you feel you must handle it for some reason or if you just feel you want to know how to master such a case, here is one way:

    You shouldn't access the read model from the command handler nor the domain when using event sourcing. However, what you could do is to use a domain service that would listen to the UserRegistered event in which you access the read model again and check whether the username still isn't a duplicate. Of course you need to use the UserGuid here as well as your read model might have been updated with the user you just created. If there is a duplicate found, you have the chance of sending compensating commands such as changing the username and notifying the user that the username was taken.

    That is one approach to the problem.

    As you probably can see, it is not possible to do this in a synchronous request-response manner. To solve that, we are using SignalR to update the UI whenever there is something we want to push to the client (if they are still connected, that is). What we do is that we let the web client subscribe to events that contain information that is useful for the client to see immediately.

    Update

    For the more complex case:

    I would say the order placement is less complex, since you can use the read model to find out if the client is valuable before you send the command. Actually, you could query that when you load the order form since you probably want to show the client that they'll get the 10% off before they place the order. Just add a discount to the PlaceOrderCommand and perhaps a reason for the discount, so that you can track why you are cutting profits.

    But then again, if you really need to calculate the discount after the order was places for some reason, again use a domain service that would listen to OrderPlacedEvent and the "compensating" command in this case would probably be a DiscountOrderCommand or something. That command would affect the Order Aggregate root and the information could be propagated to your read models.

    For the duplicate username case:

    You could send a ChangeUsernameCommand as the compensating command from the domain service. Or even something more specific, that would describe the reason why the username changed which also could result in the creation of an event that the web client could subscribe to so that you can let the user see that the username was a duplicate.

    In the domain service context I would say that you also have the possibility to use other means to notify the user, such like sending an email which could be useful since you cannot know if the user is still connected. Maybe that notification functionality could be initiated by the very same event that the web client is subscribing to.

    When it comes to SignalR, I use a SignalR Hub that the users connects to when they load a certain form. I use the SignalR Group functionality which allows me to create a group which I name the value of the Guid I send in the command. This could be the userGuid in your case. Then I have Eventhandler that subscribe to events that could be useful for the client and when an event arrives I can invoke a javascript function on all clients in the SignalR Group (which in this case would be only the one client creating the duplicate username in your case). I know it sounds complex, but it really isn't. I had it all set up in an afternoon. There are great docs and examples on the SignalR Github page.

    0 讨论(0)
  • 2021-01-29 18:50

    There is nothing wrong with creating some immediately consistent read models (e.g. not over a distributed network) that get updated in the same transaction as the command.

    Having read models be eventually consistent over a distributed network helps support scaling of the read model for heavy reading systems. But there's nothing to say you can't have a domain specific read model thats immediately consistent.

    The immediately consistent read model is only ever used to check data before issuing a command, you should never use it for directly displaying read data to a user (i.e. from a GET web request or similar). Use eventually consistent, scaleable read models for that.

    0 讨论(0)
  • 2021-01-29 18:53

    It seems to me that perhaps the aggregate is wrong here.

    In general terms, if you need to guarantee that value Z belonging to Y is unique within set X, then use X as the aggregate. X, after all, is where the invariant really exists (only one Z can be in X).

    In other words, your invariant is that a username may only appear once within the scope of all of your application's users (or could be a different scope, such as within an Organization, etc.) If you have an aggregate "ApplicationUsers" and send the "RegisterUser" command to that, then you should be able to have what you need in order to ensure that the command is valid prior to storing the "UserRegistered" event. (And, of course, you can then use that event to create the projections you need in order to do things such as authenticate the user without having to load the entire "ApplicationUsers" aggregate.

    0 讨论(0)
提交回复
热议问题