[進階]使用 Facade Pattern 取代 Model Callbacks by xdite
What is “callbacks”?
Rails 的 ActiveRecord 提供了相當方便的 callbacks,能讓開發者在寫 Controller 時,能夠寫出更加 DRY 的程式碼:
- before_crearte
- before_save
- after_create
- after_save
- …
在從前,在 Controller 裡面想要再 object 儲存之後 do_something,直觀的思路會是這樣:
class PostController def create @post = Post.new(params[:post]) @post.save @post.do_something redirect_to posts_path end end
當時的最佳模式:通常是建議開發者改用 callbacks 或者是 Observer 模式實作。避免 controller 的髒亂。
- callbacks : after_create
或者是使用 Observer
class PostController < ApplicationController def create @post = Post.new(params[:post]) @post.save redirect_to posts_path end end class PostObserver < ActiveRecord::Observer def after_create(post) post.do_something end end class Post < ActiveRecord::Base protected def do_something end end
使用 callbacks 所產生的問題
callbacks 雖然很方便,但也產生一些其他的問題。若這個 do_something 是很輕量的 db update,那這個問題還好。但如果是很 heavy 的 hit_3rd_party_api 呢?
在幾個情形下,開發者會遇到不小的麻煩。
- Model 測試:每次在測試時都會被這個 3rd_party_api 整到,因為外部通訊很慢。
- do_something_api 是很 heavy 的操作:每次寫測試還是會被很慢的 db query 整到。
- do_something_api 是很輕微的 update:但是綁定 after_save 操作,在要掃描資料庫,做大規模的某欄位修改時,會不小心觸發到不希望引發的 callbacks,造成不必要的效能問題。
當然,開發者還是可以用其他招數去閃開:
比如說若綁定 after_save 。
可以在 do_somehting 內加入對 dirty object 的偵測,避免被觸發:
def do_somthing # 資料存在,且變動的欄位包括 content if presisited? && changed.include?(“content”) the_real_thing end end
但這一招並不算理想,原因有幾:
- 每次儲存還是需要被掃描一次,可能有效能問題。
- 寫測試時還是會呼叫到可能不需要引發的 do_somehting。
- if xxx && yyy 這個 condiction chain 可能會無限延伸下去。
Facade Pattern
那麼要怎樣才能解決這個問題呢?其實我們應該用 Facade Pattern 解決這個問題。
設計模式裡面有一招 Facade Pattern,這一招其實是沒有被寫進 Design Pattern in Ruby 中的。Russ Olson 有寫了一篇文章解釋沒有收錄的原因:因為在 Ruby 中,這一招太簡單太直觀,所以不想收錄 XDDD。但他還是在網站上提供當時寫的草稿,供人參考。
What is Facade Pattern?
Facade Pattern 的目的是「將複雜的介面簡化,將複雜與瑣碎的步驟封裝起來,對外開放簡單的介面,讓客戶端能夠藉由呼叫簡單的介面而完成原本複雜的程式演算。」(來源)
延伸閱讀: (原創) 我的Design Pattern之旅[5]:Facade Pattern (OO) (Design Pattern) (C/C++)
實際舉例:
在上述的例子中,其實 do_something 有可能只會在 PostController 用到,而非所有的 model 操作都「需要」用到。所以我們 不應該將 do_somehting 丟進 callbacks(等於全域觸發),再一一寫 case 去閃避執行
與其寫在 callbacks 裡。我們更應該寫的是一個 Service Class 將這一系列複雜昂貴的行為包裝起來,以簡單的介面執行。 class PostController < ApplicationController def create CreatePostService(params[:post]) redirect_to posts_path end end class CreatePostService def self.create(params) post = Post.new(params[:post]) post.save post.do_something_a post.do_something_b post.do_something_c end end 而在寫測試,只需要對 PostCreateService 這個商業邏輯 class 寫測試即可。而 PostController 和 Post Model 就不會被殃及到。
小結
不少開發者討厭測試的原因,不只是「因為」寫測試很麻煩的原因,「跑一輪測試超級久」也是讓大家很不爽的主因之一。
其實不是這些測試框架寫的爛造成「寫測試很麻煩」、「執行測試超級久」。而是另有其他因素。
許多資深開發者逐漸意識到,真正的主因是在於目前 Rails 的 model 的設計,耦合度太高了。只要沾到 db 就慢,偏偏 db 是世界的中心。只是測某些邏輯,搞到不小心觸發其他不需要測的東西。
ActiveRecord 的問題在於,讓開發者太誤以為 ORM = model。其實開發者真正要寫的測試應該是對商業邏輯的測試,不是對 db 進行測試。
所以才會出現了用 Facade Pattern 取代 callbacks 的手法。
其他
MVC 其實有其不足的部份。坦白說,Rails 也不是真正的 MVC,而是 Model2
目前 MVC 其實是不足的,演化下來,開發者會發現 User class 裡面會開始出現這些東西:
- current_user.buy_book(book)
- current_user.add_creadit_point(point)
這屬於 User 裡面應該放的 method 嗎?well,你也可以說適合,也可以說不適合。
適合的原因是:其實你也不知道應該放哪裡,這好像是 User 執行的事,跟他有關,那就放這裡好了!不然也不知道要擺哪裡。
不適合的原因是:這是一個「商業購買行為」。不是所有人都會購物啊。這應該是一個商業購買邏輯。但是….也不知道要放在哪啊。
一直到最近,James Copelin 提出了:DCI 去補充了現有的 MVC 的不足,才算勉強解決了目前浮現的這些問題。
DCI ,與本篇談到的 Facade Pattern 算是頗類似的手法。
有關於 DCI ( Data, Context, Interaction ) 的文章,我會在之後發表。我同時也推薦各位去看這方面的主題。這個方向應該會是 Rails 專案設計上未來演化的方向之一。