Skip to content

Execercise in chapter 17.3 can be confusing #2921

Open
@videbar

Description

@videbar
  • 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:

https://doc.rust-lang.org/book/ch17-03-oo-design-patterns.html#implementing-transitions-as-transformations-into-different-types

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 from PendingReview back to Draft.
  • Require two calls to approve before the state can be changed to Published.
  • 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 to Published.

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 and PendingOneReviewPost 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 to Post.

Metadata

Metadata

Assignees

No one assigned

    Type

    No type

    Projects

    No projects

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions