fat-code-refactoring-techniques
fat-code-refactoring-techniques copied to clipboard
Earlier Draft: Service objects
Service Objects
This code demonstrates moving complex controller code into a class that manages model interactions, aka a Service Object. The controller calls the Service Object and it returns an object that encapsulates what the controller should do. This includes flash
messages, redirects, etc. The example class ControllerResponse provides a
sample of how to do this. A service object could create a presenter for the
controller to pass to the view.
After getting this reviewed, I've developed an alternative approach that focuses on keeping controller parts (setting flash, etc.) in the controller and moves other logic into the model code. See https://github.com/justin808/fat-code-refactoring-techniques/pull/12.
Service Objects Applicable Situation
- Some action involving the interaction between several models, such as:
- A customer creating an order and charging the purchase.
- Post login process for a user.
- Long complicated methods, and groups of methods on a single model.
Service Objects Example for Micro Blog
- Minors are not allowed to post profanity.
- User model counts the number of profanity words used by a minor.
- First commit presents a tangled up controller method:
def create
@micropost = current_user.microposts.build(micropost_params)
ok = true
if current_user.minor?
profanity_word = profanity_word(@micropost.content)
if profanity_word.present?
flash.now[:error] = "You cannot create a micropost with profanity: '#{profanity_word}'!"
current_user.increment(:profanity_count)
@feed_items = []
render 'static_pages/home'
ok = false
end
end
if ok
if @micropost.save
flash[:success] = "Micropost created!"
redirect_to root_url
else
@feed_items = []
render 'static_pages/home'
end
end
end
This code is not easily testable, being on the controller. The addition of the
profanity_words method on the controller also does not fit. I don't go into
building tests, as the refactor will allow easy testing.
PROFANITY_WORDS = %w(poop fart fartface poopface poopbuttface)
# Yes, this could go into a validator for the micropost, but let's suppose there's reasons
# that we don't want to do that, such as we only want to filter profanity for posts
# created by minors, etc.
# returns profanity word if existing in content, or else nil
def profanity_word(content)
words = content.split(/\W/)
PROFANITY_WORDS.each do |test_word|
puts "test_word is #{test_word}, words is #{words}"
return test_word if words.include?(test_word)
end
nil
end
- After adding MicropostCreationService:
- The code is much easier to test.
- The controller method is much simpler:
``` ruby
def create
response = MicropostCreationService.new(current_user, micropost_params[:content]).create_micropost
response.apply(self)
unless response.ok # if not ok, then we did a redirect when calling response.apply
@micropost = response.data[:micropost]
@feed_items = []
render 'static_pages/home'
end
end
```
The `response.apply` method does things like setting the flash message and
the response code.
- The main method of
MicropostCreationServiceis simple:
``` ruby
def create_micropost
micropost = @user.microposts.build(content: @content)
response = check_profanity(micropost)
unless response
response = save_micropost(micropost)
end
response
end
```
- Note how the Service utilizes the
ControllerResponseclass so as not be throwing exceptions for error scenarios.
``` ruby
def save_micropost(micropost)
if micropost.save
ControllerResponse.new(flash_message: "Micropost created!",
flash_type: :success,
redirect_path: h.root_url)
else
ControllerResponse.new(http_status: :bad_request, data: { micropost: micropost })
end
end
# return response or
def check_profanity(micropost)
if profanity_word
msg = "Whoa, better watch your language! Profanity: '#{profanity_word}' not allowed!"
@user.increment(:profanity_count)
ControllerResponse.new(flash_message: msg,
flash_type: :error,
flash_now: true, # True b/c not redirecting
http_status: :bad_request,
data: { micropost: micropost })
end
end
```
- A good test of code is how it changes when requirements change. Suppose that the code should print all the profanity in the message, and add the number of profanity words to the user's profanity counter.
- Putting the profanity calculation into it's own class further shrinks the
size of classes and clarifies the tests. The main advantage of having the
ProfanityCheckerinto its own class is that the Micropost can also use the logic for validation. - A slight error was left in the commits along with the fix. This is to show
the usefulness of having concise and simple unit tests on the
MicropostCreationService.
Service Object Solution
- Create a PORO in a
servicesdirectory (or optionally in models). - If using the services directory, be sure to add it to the load path
config.autoload_paths += %W(
#{config.root}/app/services
)
- Move all applicable methods into this class from the model.
- Use an instance of ControllerResponse to return the responses back to the controller.
- Create test in
spec/servicesdirectory.
Service Object Advantages
- Clearly bundles code around a specific, complex action.
- Promotes the creation of smaller, tighter methods when all methods in a small class are around a single purpose.
- Allows better unit testing of methods on this service object.
Service Object Disadvantages
- It can be overkill for a very simple action.
- Something simply and generally useful on the model is better put in a concern or decorator, or maybe a method class called by the model.
This service is too much coupled with the knowledge of a Rails controller (eg. flash_message, flash_type). A service should perform business logic independently from the delivery method (a controller, in this case). It should return enough informations and let the caller to take the right decision.
I'd like to share some nice to have things:
- Naming things after patterns shouldn't be preferable. Also, that service is an use case, what do you think of a name like
CreateMicropost? - Use a consistent public API across your services. Instead of
#create_micropost,#edit_micropost, use something like,#call,#run,#executeand stick with that name. - That profanity check isn't a validation, but a policy. A validation is about to the presence of an attribute, if you expect a number but receive a string, etc.. A policy is something related to the business rules of your domain: we don't want profanities in our system. I'm experimenting with policies and planning to blog about them soon.
Service Object is not a variation of a Presenter technique. They are two different things. Presenter helps out with the presentation layer, while Service layer acts as an intermediary between multiple models (and doesn't care about presentation per se). This article from Martin Fowler should shine some more light on the Service layer.
I agree with @jodosha that the pattern you are presenting is too coupled to the controller. Perhaps rename it to "Response Object", because I think there is something interesting here, but I would not confuse it with Service Objects.