在我的應用程序中,每個模型都有一個用於保存常用查詢的類(我猜它有點像DDD語言中的「Repository」)。每個這些類都通過SQLAlchemy會話對象來創建查詢。在確定某些查詢正在我的單元測試中運行的最佳方式時,我有點困難。使用無處不在的博客示例,假設我有一個包含列和屬性「日期」和「內容」的「發佈」模型。我還有一個名爲「find_latest」的「PostRepository」,它應該按照「date」降序查詢所有帖子。它看起來像這樣:Python SQLAlchemy - 嘲笑模型屬性的「desc」方法
from myapp.models import Post
class PostRepository(object):
def __init__(self, session):
self._s = session
def find_latest(self):
return self._s.query(Post).order_by(Post.date.desc())
我無法模擬Post.date.desc()調用。現在我在單元測試中爲模擬Post.date.desc的猴子修補,但我覺得可能有更好的方法。
編輯:我使用的是MOX爲模擬對象,我目前的單元測試看起來像:
import unittest
import mox
class TestPostRepository(unittest.TestCase):
def setUp(self):
self._mox = mox.Mox()
def _create_session_mock(self):
from sqlalchemy.orm.session import Session
return self._mox.CreateMock(Session)
def _create_query_mock(self):
from sqlalchemy.orm.query import Query
return self._mox.CreateMock(Query)
def _create_desc_mock(self):
from myapp.models import Post
return self._mox.CreateMock(Post.date.desc)
def test_find_latest(self):
from myapp.models.repositories import PostRepository
from myapp.models import Post
expected_result = 'test'
session_mock = self._create_session_mock()
query_mock = self._create_query_mock()
desc_mock = self._create_desc_mock()
# Monkey patch
tmp = Post.date.desc
Post.date.desc = desc_mock
session_mock.query(Post).AndReturn(query_mock)
query_mock.order_by(Post.date.desc().AndReturn('test')).AndReturn(query_mock)
query_mock.offset(0).AndReturn(query_mock)
query_mock.limit(10).AndReturn(expected_result)
self._mox.ReplayAll()
r = PostRepository(session_mock)
result = r.find_latest()
self._mox.VerifyAll()
self.assertEquals(expected_result, result)
Post.date.desc = tmp
這並不工作,但感覺醜陋,我不知道爲什麼它不能沒有「AndReturn ('test')「piece of」Post.date.desc()。andReturn('test')「
這很有道理,我對代碼的細節太在意了。感謝您的洞察力。 – 2011-03-13 19:37:13
你在說什麼,而不是做一個單元測試(一個邏輯單元),做一個集成測試(與數據庫)。這是一種有效的方法,可能對於ORM來說最爲明智,但這些測試的性能會受到一兩個量級的影響。對? – 2014-12-10 06:53:04
你是正確的,編寫一個能以某種方式訪問較慢資源(不管是磁盤驅動器,外部服務還是數據庫)的測試(無論你想稱之爲「單元」測試還是「集成」測試)將意味着比不訪問這種資源的測試慢。 – 2014-12-10 15:58:36