Description
- I have checked the latest
main
branch to see if this has already been fixed - I have searched existing issues and pull requests for duplicates
URL to the section(s) of the book with this problem:
Description of the problem:
During chapter 17.3 it is explained how the same system could be implemented first using the state design pattern and then a variation of this pattern that takes advantage of some of rust features. The system being explained is a blogpost that can be in different states: draft, pending review and post.
It is explained how to implement the blogpost using the state design pattern and a simple exercise is left for the reader:
- Add a
reject
method that changes the post’s state fromPendingReview
back toDraft
.- Require two calls to
approve
before the state can be changed toPublished
.- Allow users to add text content only when a post is in the
Draft
state. Hint: have the state object responsible for what might change about the content but not responsible for modifying the Post.
Which is great.
Then it is explained how this pattern can be modified to use rust features and at the end of the chapter it says:
Try the tasks suggested for additional requirements that we mentioned at the start of this section on the
blog
crate as it is after Listing 17-20 to see what you think about the design of this version of the code.
I think this is a problem because, at least to the best of my knowledge, there's no easy way no implement the second requirement in the new design:
Require two calls to
approve
before the state can be changed toPublished
.
I found this post that explains the problems with this exercise and the best solution states:
I think the idea that they are getting at is that you add another type, so that you have
PendingTwoReviewsPost
andPendingOneReviewPost
or the like, and approving the former returns the latter, and approving the latter returns a Post. I.e., encoding the state into the types.
which seems to be the only way to solve the problem without modifying how the API is called (which I think is the intention). If this is indeed the intended solution, it's awful, if I wanted to change the number of required approvals from 2 to let's say 5, instead of changing a parameter in the script, I would need to define 3 new structs!
Suggested fix:
Replace the line:
Try the tasks suggested for additional requirements that we mentioned at the start of this section on the
blog
crate as it is after Listing 17-20 to see what you think about the design of this version of the code.
with something like:
This design has other problems however, if you try the tasks suggested for additional requirements, you will realize that there is no easy way to change the number of approvals required for a post to change from
PendingReview
toPost
.