-
Notifications
You must be signed in to change notification settings - Fork 526
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
[Question] Domain Design of Collaboration System in Epilogue #305
Comments
Hi @gregbrowndev . It seems to me that the An alternative is for the Publication to expose methods for mutating state.
Lastly, it's worth noting that not everything has to be written this way. If the draft approval and revision process is truly generic, then you might as well just use Django CRUD and keep it simple. Save the fancy modelling for the parts of your system that make your domain special and complex. |
Hi @bobthemighty, I greatly appreciate your reply.
Publication does act as a root for the set of Revisions in that it has a global identifier that other non-publisher users use to find the live (most recent) revision (they are not aware of the draft but can see revision history). This ID doesn't change when new revisions are published. Do you think it is still better suited as a repo.?
The data that the publishers submit is fairly simple, it is just a handful of fields/metadata describing their upload. E.g. class TimetableDataset(Revision):
url: str
filename: str
admin_areas_names: List[str]
publisher_creation_datetime: Optional[datetime.datetime]
first_expiring_service: Optional[datetime.datetime]
class GPSFeedDataset(Revision):
url: str
username: str
password: str
requestor_ref: str
is_capability_request: Optional[bool]
class FaresDataset(Revision):
...
class DisruptionDataset(Revision):
... The main problem that we're trying to get away from is that this class hierarchy is currently all stored in the same Django model via single table inheritance. Having grown organically this way alongside the revision functionality, it's now virtually impossible to change, as all the views and test code are highly coupled to this one model. And, as it isn't even the main queryable root model, this precludes the possibility of using something like Django's proxy model to easily separate out data and behaviours. We basically have the same problem that you describe in the epilogue, where our domain model is inexpressive as it is forced to mirror the data model. In Django dot syntax it looks like:
If I were to treat the different Revision subclasses as Value Objects, or perhaps have the Revision compose the Dataset (since there are fields on the Revision the users shouldn't change, e.g. published_by) and treat the Dataset as a Value Object, then would it be OK to pass those in/out of the aggregate? Again, I am grateful for any help. |
Hey @gregbrowndev 👋 how are you? I'm interested in this topic as I have something similar. But first a clarifying question. You wrote
Do you mean
so this is where there's a table for revision and each revision can be a GPSFeedDataset, TimetableDataset, etc. the common fields are in the revision table, and those fields unique to individual model are in separate child tables
|
Hey @simkimsia, I'm well thanks. Hope you are too. (And to note, I don't mean to follow you around 🙈) To answer your question, yes it is the second one! The two tables in particular in the DB are Ignoring the fact the table is sparse / has poor data integrity, it wouldn't be that much of an issue if we could get it back into a nice structure. Partly the reason we ended up with this was to avoid adding even more levels of traversal to get to the important data, (i.e. SQLAlchemy has support for Single Table Inheritance that I'm considering for my mapper layer. (It is possibly a bit bonkers using Django ORM and SQLAlchemy as a mapper side-by-side, but anyhow...) |
BTW @simkimsia, I tried out the classic data mapper approach with SQLAlchemy alongside my Django app. It definitely would work quite elegantly but it seemed to have compatibility issues with mapping onto the Pydantic models. It probably wouldn't be too much of a loss to remove Pydantic, as it says in Appendix E, the validation can be done at the edges. However, SQLAlchemy manages its own DB session, of course, and completely bypasses Django's ORM, so I don't know how many would feel about that at work. So I've decided to put it aside and implement the mappers imperatively with the Django ORM. |
hahaha but i did recommend cosmicpython to you. Hope you find it as useful as I do! 🙌 Anyway, I have other clarifying questions I like to ask. Question 1
So are you saying pydantic was overkill or totally didn't work in Appendix E or something else? In a weird coincidence I was doing my own research into the chapters 8, 9, 10, 11 of cosmicpython and ended up asking pydantic abt appendix E as well. Comparing pydantic against schema which was recommended in Appendix E by the authors as a way for validation. The question to pydantic is posted here pydantic/pydantic#1486 Question 2Have you tried django typed models? https://github.com/craigds/django-typed-models it allows single table inheritance in Django tho it was last updated last year May Question 3
when you manage to get it to work eventually, regardless if u use django orm imperatively with the mappers, or something else, can share with me your eventual architectural decision and snippets of the implementation? I am curious myself Thank you 🙏 |
@simkimsia yes, it has been super helpful! 🙏
To be honest, I haven't fully digested all the material in Appendix E yet or thought too much about validation. I am still using Pydantic in all my entities, as I like that it works nicely with DryPython's use case contracts. I didn't want to lose that functionality in favour of SQLAlchemy (since that brings its own substantial complications).
That discussion is actually really helpful. Up to now, I based all my entities around Pydantic just because of the story contract functionality but didn't know if doing so was a best practice or not. So I will be going forward with Pydantic but I'll let you know if I hit any blockers.
Cheers for the suggestion! I haven't used it but I have looked into it. That will be something I will add in time, once things are decoupled from the model and there isn't a greater risk of breaking things.
Yeah of course! I can send you some snippets over email if you drop me a line. All the code I am working on will eventually be open-sourced on GitHub too but our client hasn't organised that yet. |
Hi there,
First, I want to thank you for your book. It has been a tremendous help in my journey so far in learning DDD. I am beginning to apply what I've learned to a Django project that has caused me much pain over the last 2 years. Many of the larger concepts have started to click into place, which feels amazing. However, I am struggling with the basic domain modelling of my problem.
The only reason I ask such for specific help is that it is similar to the document collaboration system that you talk about in the epilogue.
The app involves allowing transport authorities to publish different types of datasets via basic file uploads. Each dataset type may have different rules and processes, e.g. validation. However, the part that I'm having the most difficulty with is the generic CMS functionality that you would find in any off the shelf CMS.
In brief, the publisher creates a 'draft' version of their dataset which must be submitted, validated and then reviewed before it can be published. And, to edit a published dataset, a new draft must be submitted, reviewed, etc. There can only be a single draft for review at a time, and the user(s) can edit the draft any number of times (though note: real-time collaboration isn't the purpose of the app). Once a version has been published, it cannot be modified.
The challenge I'm having is in defining what the aggregate should look like, where the responsibility should lie for how the client creates and edits the revisions, and how to make the root responsible for these invariants.
The code snippet below defines a 'Publication' aggregate. Partly the reason I'm not 100% confident is that as it states in blue book (in the chapter about factories):
but it also says:
so I can see the
start_revision
method is like a factory method to expand the aggregate, but it still feels quite awkward that the client has to pass back and forth the revision objects and thus be aware of the Publication's internal structure.Do you have any tips on how I can improve my approach to tackling this design problem? Or how you dealt with Document version tracking from the project that you worked on? Thank you for any help you may offer.
The text was updated successfully, but these errors were encountered: