I have an Agenda
that holds many Card
, and a Card
has multiple DetailItem
that hold a value like Email
, Phone
and a Label
So, I can do:
agenda = new Agenda()
oneCard = new Card()
item = new DetailItem(new Email("[email protected]"), new Label("Work")
oneCard.addItem(item)
agenda.addCard(oneCard)
By rule, there can only be one DetailItem
with an Email
instance with "[email protected]" value, so if you try to add a new item with that data it will raise an exception.
This seems to be ok until I try to update a DetailItem
. I can't find a way that I feel comfortable with.
I try to think in terms of business model and not implementation details, but I cannot leave them off for a long, and they enter the domain I like it or not.
The thing is that I will have a REST interface, and I have two ways of doing things.
- Send a PUT to
/cards/<cardId>
with adetailItems' array, fetch the
Cardby ID, create a new
Cardwith the new data provided in the PUT, and sync the current
Card` with the new one. - Send a PUT to /cards//items/, fetch the
Card
, find theDetailItem
, and update it the same way in option 1
If I go with option 1, I have to remove all DetailItem
from current Card
that don't exists in the new one. That leads with some INSERT queries produced by the ORM, and some DELETES. No UPDATE at all.
If I go with option 2, I will have many PUTs to change multiple items, which is not performant at all, and it will lead me to introduce an ID field to the DetailItem
so I can identify them, which introduces something is not part of the domain!.
The only option I found was to go with option 1, and send a request like this:
{
...
detailItems: [
{
type: "email",
oldValue: "[email protected]",
oldLabel: "Work",
newValue: "[email protected]",
newLabel: "Work",
}
]
}
So I can do something like this:
card = agenda.getCardIdentifiedBy(the_identifier)
for(itemUpdate in jsonData.detailItems)
itemClass = ValueClassMapper.from(itemUpdate.type) // "email" -> Email
oldItem = new DetailItem(new itemClass(itemUpdate.oldValue), new Label(itemUpdate.oldLabel))
newItem = new DetailItem(new itemClass(itemUpdate.newValue), new Label(itemUpdate.newLabel))
card.updateItemWith(oldItem, newItem)
But, I don't know, maybe I have a wrong abstraction Can anyone help? Thanks in advance!
Part of the problem that you are running into here is that HTTP remote authoring semantics are at odds with the notion of an autonomous domain model.
HTTP PUT means "make your copy of this document look like my copy of this document". It is analogous to "save file", or "commit these rows to the database". The semantics of the request fit the model of an anemic data store.
On the other hand, our domain models are typically autonomous state machines, that expect to decide for themselves what state they should transition into based on new information. In other words, all domain model interactions roughly follow the following pattern:
The model then changes its own data structures in response to this information (with the application code responsible for parsing the new information into a data structure that the domain model will understand).
This doesn't mean that you can't use HTTP and REST with domain models; but it does mean that the resource model looks very different from what you have described here.
The most straightforward approach would be to POST the new information to your agenda resource (autonomous domain entities really don't want you trying to directly modify their internals).
It would then be your application's responsibility to parse the body of the request, and then to pass that parsed representation along to your agenda domain entity, the domain entity would then decide how/if to integrate the new information with the information that it already knows about.
Another approach would be to treat the new information as a new document to be stored, where each new document would have its own URI:
Where the changes to
/agenda/12345
are a side effect of storing the message -- see Webber 2011. Please be aware that this approach can make cache invalidation more challenging.