Hacker Newsnew | past | comments | ask | show | jobs | submit | stemmlerjs's commentslogin

Thanks for this! Good resource. The AWS Docs are way too massive.


"There's much less chances for things to go wrong in this design since there is only one code path for updating the components state" ^ - well said! I'm going to update the article to be a bit more clear about that.


That's what nightmares are made of


After further thought, because overarching opinionated frameworks sometimes aren't the answer.

From an architectural POV, sometimes frameworks dont solve all your problems, and sometimes they make things harder for you by going against the grain of how your problem domain actually wants to be architected.

Knowing the basics of good software design and how to not use a framework if you're not able to has a lot of merit.

Some domain driven development can actually be idealized as you building your own framework of the problem domain. Why complicate that by adding an additional layer of infrastructure?

It really does depend on the project and evaluating the framework.


Object modeling is definitely not the easiest thing in the world, but for complex problem domains, it can be what makes or breaks being able to continuously push a project forward.

For simple RESTful todo apps, I'd definitely say you could get away without objects.

But if I was to build a GitLab clone, it would be incredibly difficult to model such a complex problem domain without using a domain model.


I've been hearing a lot about NestJS recently. I've yet to try it out on a Greenfield project.

Yeah, I'm totally with you on that. Frameworks have the structure we all need but seem to neglect:) Although there is a lot of merit in learning by doing (hence the purpose of the article and my website overall).

Lots of brownfield vanilla Express.js Node APIs can be improved using this pattern.


The point of DDD is to encapsulate your domain models to keep the invariants satisfied. How can you restrict the very possibility of creating invalid instances of a User this way? There's nothing stopping me from using the "new" keyword and making an invalid user with a name 400 characters long.

This is why we use private constructors. The Value Object is a place to encapsulate that domain logic.

The private constructor removes that very possibility and ensures the only way to create a User is to use the static factory method, which validates the User.

I admit, when I first saw the layers of encapsulation, it was a bit off to me. But that's a small price to pay for model integrity.


> How can you restrict the very possibility of creating invalid instances of a User this way? There's nothing stopping me from using the "new" keyword and making an invalid user with a name 400 characters long.

Why do you need to restrict this possibility? How many places in your code create new users, and how many places in your code need to validate users? If the answer is “many,” you have serious design problems that should not be solved by introducing more classes but by restricting how many different areas of code do the same job.

Also, the private constructors do absolutely nothing. The static factory methods in this blog post could be written the exact same way as a constructor, and they would work exactly the same. The only reason the author uses a static factory method is because that's what he was taught to do in Java.

It's just OOP for OOP's sake in a language with tons of features designed for avoiding OOP.


> How many places in your code create new users, and how many places in your code need to validate users? If the answer is “many,” you have serious design problems that should not be solved by introducing more classes but by restricting how many different areas of code do the same job.

Consider an application where you can edit and add users. Consider that you can also add users to groups and remove users from groups. Consider that you might also be able to add policies to users. These are a few different areas in code where you might need to create an instance of a User.

This is exactly what we're doing; we're limiting the amount of places where valid users can get created. The User domain object and it's invariants are validated right here at the model.

If there was an invariant saying that users can't have more than 5 policies on them, a user.addPolicy(policy) method would exist right on the user model/aggregate root, and the aggregate root would be responsible for determining if any invariants will be unsatisfied after the operation.

> Also, the private constructors do absolutely nothing. The static factory methods in this blog post could be written the exact same way as a constructor, and they would work exactly the same. The only reason the author uses a static factory method is because that's what he was taught to do in Java.

The reason why I use a static factory method is because it allows you to protect the aggregate invariants (domain logic that deems the model to be correct). The language has nothing to do with it.

If I was to create Coordinates, I need to ensure that the Latitude and Longitude fulfill the domain requirements, and I want to be able to know if I was able to create that object or not.

This approach usually means using a Result class to handle errors (https://enterprisecraftsmanship.com/2015/03/20/functional-c-...).

I'll be writing another article on that topic.

This is a different way of programming, for sure. DDD takes a bit of time to wrap your head around. But when projects are sufficiently complex, it pays off to know that there's no possibility for an invalid aggregate to be floating around at runtime.

> It's just OOP for OOP's sake in a language with tons of features designed for avoiding OOP.

Could not be more incorrect.

If you're interested in learning more about DDD, check out "DDD Distilled" for a quick primer on it and why the chosen patterns are used. There's a method to the madness.

Think of it as like building a Domain-Specific language that any of the 4 developers in your company that need to write code will be able to use without breaking any rules of the domain. If it's just you writing code on a project that is relatively non-complex, DDD is probably not the optimal choice.


Well you could use the Python approach of "we are all adults" and allow people to do the wrong thing if they are explicitly asking to do it.

I do believe it's better to be strict but allow a loophole just in case future use requires it.

For example in my codebase, only valid objects can be created via User.create but we do create invalid ones for testing since it's not necessary to have full objects there.


That's pretty funny. I like that. Yeah, I'd say it definitely depends on your project and your team.

There are also ways to opt-in and out of following rules like this, even from a TypeScript context- you just have to be more aggressive about it.

For example, you could access private methods by doing:

User['create']() for example. Sometimes this is necessary for testing.


Wow, nice work! Starred it to dig around your repo my next lunch break.


Love this response. Very well said.


I appreciate that. Thank you!


Guidelines | FAQ | Lists | API | Security | Legal | Apply to YC | Contact

Search: