UnitTest Framework - Hızlı Kılavuz
Birim testi, işlevler, yöntemler ve sınıf gibi ayrı ayrı kaynak kod birimlerinin kullanıma uygun olup olmadıklarını belirlemek için test edildiği bir yazılım test yöntemidir. Sezgisel olarak, bir ünite bir uygulamanın test edilebilir en küçük parçası olarak görülebilir. Birim testleri, geliştirme sürecinde programcılar tarafından oluşturulan kısa kod parçalarıdır. Bileşen testinin temelini oluşturur.
Birim testi aşağıdaki iki şekilde yapılabilir -
Manuel Test | Otomatik Test |
---|---|
Test senaryolarının herhangi bir araç desteği olmadan manuel olarak yürütülmesi manuel test olarak bilinir.
|
Araç desteğinin alınması ve test durumlarının otomasyon aracı kullanılarak yürütülmesi, otomasyon testi olarak bilinir.
|
JUnit, Java programlama dili için bir birim test çerçevesidir. JUnit, test güdümlü geliştirmenin geliştirilmesinde önemli olmuştur ve JUnit ile ortaya çıkan ve topluca xUnit olarak bilinen birim test çerçeveleri ailesinden biridir. JUnit Eğitimini buradan öğrenebilirsiniz .
Bazen "PyUnit" olarak adlandırılan Python birim testi çerçevesi, Kent Beck ve Erich Gamma tarafından geliştirilen JUnit'in Python dili sürümüdür. PyUnit, Python sürüm 2.1'den itibaren Python Standart Kitaplığı'nın bir parçasını oluşturur.
Python birim test çerçevesi, test otomasyonunu, testler için kurulum ve kapatma kodunun paylaşılmasını, testlerin koleksiyonlarda toplanmasını ve testlerin raporlama çerçevesinden bağımsızlığını destekler. Unittest modülü, bir dizi test için bu nitelikleri desteklemeyi kolaylaştıran sınıflar sağlar.
Bu eğitim, yeni başlayanların Python test çerçevesinin temel işlevlerini anlamalarına yardımcı olmak için hazırlanmıştır. Bu öğreticiyi tamamladıktan sonra, kendinizi sonraki seviyelere götürebileceğiniz Python test çerçevesini kullanma konusunda orta düzeyde bir uzmanlık düzeyinde bulacaksınız.
Python dilini kullanarak yazılım geliştirmede makul uzmanlığa sahip olmalısınız. Bizim Python öğretici Python öğrenmeye başlamak için iyi bir yerdir. Yazılım testinin temelleri hakkında bilgi sahibi olmak da arzu edilir.
Ortam Kurulumu
Testleri yazmak için gerekli sınıflar 'unittest' modülünde bulunmalıdır. Python'un eski sürümlerini kullanıyorsanız (Python 2.1'den önce), modül şuradan indirilebilir:http://pyunit.sourceforge.net/. Ancak, unittest modülü artık standart Python dağıtımının bir parçasıdır; bu nedenle ayrı bir kurulum gerektirmez.
'unittest', test otomasyonunu, testler için kurulum ve kapatma kodunun paylaşılmasını, testlerin koleksiyonlarda toplanmasını ve testlerin raporlama çerçevesinden bağımsızlığını destekler.
Unittest modülü, bir dizi test için bu nitelikleri desteklemeyi kolaylaştıran sınıflar sağlar.
Bunu başarmak için, unittest aşağıdaki önemli kavramları destekler -
test fixture- Bu, bir veya daha fazla test gerçekleştirmek için gereken hazırlığı ve tüm ilişkili temizleme eylemlerini temsil eder. Bu, örneğin, geçici veya proxy veritabanları, dizinler oluşturmayı veya bir sunucu işlemini başlatmayı içerebilir.
test case- Bu, en küçük test birimidir. Bu, belirli bir girdi setine özel bir yanıt olup olmadığını kontrol eder. unittest bir temel sınıf sağlar,TestCase, yeni test senaryoları oluşturmak için kullanılabilir.
test suite- Bu, test senaryolarının, test gruplarının veya her ikisinin bir koleksiyonudur. Bu, birlikte yürütülmesi gereken testleri bir araya getirmek için kullanılır. Test paketleri TestSuite sınıfı tarafından uygulanır.
test runner- Bu, testlerin yürütülmesini düzenleyen ve sonucu kullanıcıya sağlayan bir bileşendir. Koşucu, testlerin yürütülmesinin sonuçlarını belirtmek için bir grafik arayüz, metinsel bir arayüz kullanabilir veya özel bir değer döndürebilir.
Birim Testi Oluşturma
Aşağıdaki adımlar, basit bir birim testi yazmakla ilgilidir -
Step 1 - Unittest modülünü programınıza aktarın.
Step 2- Test edilecek bir işlev tanımlayın. Aşağıdaki örnekte, add () işlevi teste tabi tutulacaktır.
Step 3 - Unittest.TestCase'i alt sınıflara ayırarak bir test olayı oluşturun.
Step 4- Sınıf içinde bir yöntem olarak bir testi tanımlama. Yöntemin adı 'test' ile başlamalıdır.
Step 5- Her test çağrısı, TestCase sınıfının işlevini onaylar. Pek çok iddia türü vardır. Aşağıdaki örnek assertEquals () işlevini çağırır.
Step 6 - assertEquals () işlevi, add () işlevinin sonucunu arg2 bağımsız değişkeniyle karşılaştırır ve karşılaştırma başarısız olursa assertionError atar.
Step 7 - Son olarak, unittest modülünden main () yöntemini çağırın.
import unittest
def add(x,y):
return x + y
class SimpleTest(unittest.TestCase):
def testadd1(self):
self.assertEquals(add(4,5),9)
if __name__ == '__main__':
unittest.main()
Step 8 - Yukarıdaki komut dosyasını komut satırından çalıştırın.
C:\Python27>python SimpleTest.py
.
----------------------------------------------------------------------
Ran 1 test in 0.000s
OK
Step 9 - Aşağıdaki üçü bir testin olası sonuçları olabilir -
Sr.No | Mesaj ve Açıklama |
---|---|
1 | OK Test başarılı. Konsolda 'A' görüntülenir. |
2 | FAIL Test başarılı olmaz ve bir AssertionError istisnası oluşturur. Konsolda 'F' görüntülenir. |
3 | ERROR Test, AssertionError dışında bir istisna yaratır. Konsolda 'E' görüntülenir. |
Bu sonuçlar konsolda sırasıyla ".", "F" ve "E" ile görüntülenir.
Komut satırı arayüzü
Unittest modülü komut satırından tekli veya çoklu testleri çalıştırmak için kullanılabilir.
python -m unittest test1
python -m unittest test_module.TestClass
python -m unittest test_module.TestClass.test_method
unittest aşağıdaki komut satırı seçeneklerini destekler. Tüm komut satırı seçeneklerinin bir listesi için aşağıdaki komutu kullanın -
Python –m unittest -h
Sr.No | Seçenek ve Açıklama |
---|---|
1 | -h, --help Bu mesajı göster |
2 | v, --verbose Ayrıntılı çıktı |
3 | -q, --quiet Minimum çıktı |
4 | -f, --failfast İlk başarısızlıkta dur |
5 | -c, --catch Control-C'yi yakalayın ve sonuçları görüntüleyin |
6 | -b, --buffer Test çalıştırmaları sırasında tampon stdout ve stderr |
Bu bölüm, unittest modülünde tanımlanan sınıfları ve yöntemleri tartışır. Bu modülde beş ana sınıf vardır.
TestCase Sınıfı
Bu sınıfın amacı, test edilebilir en küçük birimi temsil eder. Test rutinlerini tutar ve her rutini hazırlamak ve daha sonra temizlemek için kancalar sağlar.
Aşağıdaki yöntemler TestCase sınıfında tanımlanmıştır -
Sr.No | Yöntem ve Açıklama |
---|---|
1 | setUp() Test fikstürünü hazırlamak için yöntem çağrıldı. Bu, test yöntemini çağırmadan hemen önce çağrılır |
2 | tearDown() Yöntem, test yöntemi çağrıldıktan ve sonuç kaydedildikten hemen sonra çağrılır. Test yöntemi bir istisna oluştursa bile bu çağrılır, |
3 | setUpClass() Tek bir sınıf çalıştırmasında testlerden önce çağrılan bir sınıf yöntemi. |
4 | tearDownClass() Tek bir sınıftaki testler çalıştırıldıktan sonra çağrılan bir sınıf yöntemi. |
5 | run(result = None) Olarak geçirilen test sonucu nesnesine sonucu toplayarak, testi uygulama sonucu . |
6 | skipTest(reason) Bunun bir test yöntemi veya setUp () sırasında çağrılması mevcut testi atlar. |
7 | debug() Sonucu toplamadan testi çalıştırın. |
8 | shortDescription() Testin tek satırlık bir açıklamasını döndürür. |
Fikstür
Bir TestCase sınıfı içinde yazılmış çok sayıda test olabilir. Bu test yöntemlerinin başlatılması için veritabanı bağlantısı, geçici dosyalar veya diğer kaynaklar gerekebilir. Bunlara demirbaş denir. TestCase, testlerinizin ihtiyaç duyduğu tüm armatürleri yapılandırmak ve temizlemek için özel bir kanca içerir. Fikstürleri yapılandırmak için setUp () öğesini geçersiz kılın. Temizlemek için tearDown () öğesini geçersiz kılın.
Aşağıdaki örnekte, TestCase sınıfının içine iki test yazılmıştır. İki değerin toplanması ve çıkarılmasının sonucunu test ederler. Setup () yöntemi, argümanları her testin shortDescription () yöntemine göre başlatır. teardown () yöntemi her testin sonunda çalıştırılacaktır.
import unittest
class simpleTest2(unittest.TestCase):
def setUp(self):
self.a = 10
self.b = 20
name = self.shortDescription()
if name == "Add":
self.a = 10
self.b = 20
print name, self.a, self.b
if name == "sub":
self.a = 50
self.b = 60
print name, self.a, self.b
def tearDown(self):
print '\nend of test',self.shortDescription()
def testadd(self):
"""Add"""
result = self.a+self.b
self.assertTrue(result == 100)
def testsub(self):
"""sub"""
result = self.a-self.b
self.assertTrue(result == -10)
if __name__ == '__main__':
unittest.main()
Yukarıdaki kodu komut satırından çalıştırın. Aşağıdaki çıktıyı verir -
C:\Python27>python test2.py
Add 10 20
F
end of test Add
sub 50 60
end of test sub
.
================================================================
FAIL: testadd (__main__.simpleTest2)
Add
----------------------------------------------------------------------
Traceback (most recent call last):
File "test2.py", line 21, in testadd
self.assertTrue(result == 100)
AssertionError: False is not true
----------------------------------------------------------------------
Ran 2 tests in 0.015s
FAILED (failures = 1)
Sınıf Fikstürü
TestCase sınıfı, bir TestCase sınıfı içinde tek tek testlerin yürütülmesinden önce yürütülecek şekilde geçersiz kılınabilen bir setUpClass () yöntemine sahiptir. Benzer şekilde, sınıftaki tüm testlerden sonra tearDownClass () yöntemi çalıştırılacaktır. Her iki yöntem de sınıf yöntemleridir. Bu nedenle, @classmethod yönergesi ile dekore edilmeleri gerekir.
Aşağıdaki örnek, bu sınıf yöntemlerinin kullanımını gösterir -
import unittest
class TestFixtures(unittest.TestCase):
@classmethod
def setUpClass(cls):
print 'called once before any tests in class'
@classmethod
def tearDownClass(cls):
print '\ncalled once after all tests in class'
def setUp(self):
self.a = 10
self.b = 20
name = self.shortDescription()
print '\n',name
def tearDown(self):
print '\nend of test',self.shortDescription()
def test1(self):
"""One"""
result = self.a+self.b
self.assertTrue(True)
def test2(self):
"""Two"""
result = self.a-self.b
self.assertTrue(False)
if __name__ == '__main__':
unittest.main()
TestSuite Sınıfı
Python'un test çerçevesi, test senaryosu örneklerinin test ettikleri özelliklere göre bir arada gruplandırılabileceği yararlı bir mekanizma sağlar. Bu mekanizma, unittest modülünde TestSuite sınıfı tarafından kullanıma sunulmuştur.
Aşağıdaki adımlar, bir test paketinin oluşturulması ve çalıştırılmasıyla ilgilidir.
Step 1 - TestSuite sınıfının bir örneğini oluşturun.
suite = unittest.TestSuite()
Step 2 - Paketteki bir TestCase sınıfının içine testler ekleyin.
suite.addTest(testcase class)
Step 3 - Bir sınıftan testler eklemek için makeSuite () yöntemini de kullanabilirsiniz
suite = unittest.makeSuite(test case class)
Step 4 - Süite bireysel testler de eklenebilir.
suite.addTest(testcaseclass(""testmethod")
Step 5 - TestTestRunner sınıfının bir nesnesini oluşturun.
runner = unittest.TextTestRunner()
Step 6 - Paketteki tüm testleri çalıştırmak için run () yöntemini çağırın
runner.run (suite)
Aşağıdaki yöntemler TestSuite sınıfında tanımlanmıştır -
Sr.No | Yöntem ve Açıklama |
---|---|
1 | addTest() Test paketine bir test yöntemi ekler. |
2 | addTests() Birden çok TestCase sınıfından testler ekler. |
3 | run() Bu paketle ilişkili testleri çalıştırarak sonucu test sonucu nesnesinde toplar |
4 | debug() Sonucu toplamadan bu paketle ilişkili testleri çalıştırır. |
5 | countTestCases() Bu test nesnesiyle temsil edilen testlerin sayısını verir |
Aşağıdaki örnek, TestSuite sınıfının nasıl kullanılacağını gösterir -
import unittest
class suiteTest(unittest.TestCase):
def setUp(self):
self.a = 10
self.b = 20
def testadd(self):
"""Add"""
result = self.a+self.b
self.assertTrue(result == 100)
def testsub(self):
"""sub"""
result = self.a-self.b
self.assertTrue(result == -10)
def suite():
suite = unittest.TestSuite()
## suite.addTest (simpleTest3("testadd"))
## suite.addTest (simpleTest3("testsub"))
suite.addTest(unittest.makeSuite(simpleTest3))
return suite
if __name__ == '__main__':
runner = unittest.TextTestRunner()
test_suite = suite()
runner.run (test_suite)
MakeSuite () yöntemini kullanarak satırları ve yorum ifadesini kaldırarak addTest () yöntemini deneyebilirsiniz.
TestLoader Sınıfı
Unittest paketi, sınıflardan ve modüllerden test takımları oluşturmak için kullanılan TestLoader sınıfına sahiptir. Varsayılan olarak, unittest.defaultTestLoader örneği, unittest.main (0 yöntemi çağrıldığında otomatik olarak oluşturulur. Açık bir örnek, ancak belirli özelliklerin özelleştirilmesini sağlar.
Aşağıdaki kodda, TestLoader nesnesi kullanılarak iki sınıftaki testler bir List'de toplanır.
import unittest
testList = [Test1, Test2]
testLoad = unittest.TestLoader()
TestList = []
for testCase in testList:
testSuite = testLoad.loadTestsFromTestCase(testCase)
TestList.append(testSuite)
newSuite = unittest.TestSuite(TestList)
runner = unittest.TextTestRunner()
runner.run(newSuite)
Aşağıdaki tablo, TestLoader sınıfındaki yöntemlerin bir listesini gösterir -
S.No | Yöntem ve Açıklama |
---|---|
1 | loadTestsFromTestCase() Bir TestCase sınıfında bulunan tüm test vakalarının bir paketini döndür |
2 | loadTestsFromModule() Verilen modülde bulunan tüm test durumlarının bir paketini döndür. |
3 | loadTestsFromName() Bir dizge tanımlayıcısı verilen tüm test durumlarının bir paketini döndürür. |
4 | discover() Belirtilen başlangıç dizininden alt dizinlerde yineleyerek tüm test modüllerini bulun ve bir TestSuite nesnesi döndürün |
TestResult Sınıfı
Bu sınıf, başarılı olan testler ve başarısızlıkla karşılaşan testler hakkında bilgi toplamak için kullanılır. Bir TestResult nesnesi, bir dizi testin sonuçlarını depolar. TestResult örneği, TestRunner.run () yöntemi tarafından döndürülür.
TestResult örnekleri aşağıdaki özniteliklere sahiptir -
Sr.No | Öznitelik ve Açıklama |
---|---|
1 | Errors TestCase örneklerinin 2 demetini ve biçimlendirilmiş izleme boşluklarını tutan dizeleri içeren bir liste. Her demet, beklenmedik bir istisna oluşturan bir testi temsil eder. |
2 | Failures TestCase örneklerinin 2 demetini ve biçimlendirilmiş izleme boşluklarını tutan dizeleri içeren bir liste. Her demet, bir hatanın TestCase.assert * () yöntemleri kullanılarak açıkça bildirildiği bir testi temsil eder. |
3 | Skipped TestCase örneklerinin 2 demetini ve testi atlama nedenini içeren dizeleri içeren bir liste. |
4 | wasSuccessful() Şimdiye kadar çalıştırılan tüm testler geçtiyse True döndür, aksi takdirde False döndürür. |
5 | stop() Bu yöntem, çalıştırılmakta olan testler setinin iptal edilmesi gerektiğini belirtmek için çağrılabilir. |
6 | startTestRun() Herhangi bir test yürütülmeden önce bir kez çağrılır. |
7 | stopTestRun() Tüm testler yapıldıktan sonra bir kez çağrılır. |
8 | testsRun Şimdiye kadar yapılan toplam test sayısı. |
9 | Buffer True olarak ayarlanmışsa, sys.stdout ve sys.stderr çağrılan startTest () ve stopTest () arasında arabelleğe alınacaktır. |
Aşağıdaki kod bir test paketini yürütür -
if __name__ == '__main__':
runner = unittest.TextTestRunner()
test_suite = suite()
result = runner.run (test_suite)
print "---- START OF TEST RESULTS"
print result
print "result::errors"
print result.errors
print "result::failures"
print result.failures
print "result::skipped"
print result.skipped
print "result::successful"
print result.wasSuccessful()
print "result::test-run"
print result.testsRun
print "---- END OF TEST RESULTS"
Çalıştırıldığında kod aşağıdaki çıktıyı görüntüler -
---- START OF TEST RESULTS
<unittest.runner.TextTestResult run = 2 errors = 0 failures = 1>
result::errors
[]
result::failures
[(<__main__.suiteTest testMethod = testadd>, 'Traceback (most recent call last):\n
File "test3.py", line 10, in testadd\n
self.assertTrue(result == 100)\nAssert
ionError: False is not true\n')]
result::skipped
[]
result::successful
False
result::test-run
2
---- END OF TEST RESULTS
Python test çerçevesi, belirli bir koşulu test eden Python'un yerleşik assert () işlevini kullanır. İddia başarısız olursa, bir AssertionError ortaya çıkar. Test çerçevesi daha sonra testi Başarısız olarak tanımlayacaktır. Diğer istisnalar Hata olarak değerlendirilir.
Unittest modülünde aşağıdaki üç onaylama işlevi kümesi tanımlanmıştır -
- Temel Boolean İddiaları
- Karşılaştırmalı İddialar
- Koleksiyonlar için Onaylar
Temel onaylama işlevleri, bir işlemin sonucunun Doğru mu Yanlış mı olduğunu değerlendirir. Tüm iddia yöntemleri kabul edermsg belirtilirse, başarısızlık durumunda hata mesajı olarak kullanılan argüman.
Sr.No | Yöntem ve Açıklama |
---|---|
1 | assertEqual(arg1, arg2, msg = None) Arg1 ve arg2'nin eşit olduğunu test edin . Değerler eşit değilse, test başarısız olur. |
2 | assertNotEqual(arg1, arg2, msg = None) Arg1 ve arg2'nin eşit olmadığını test edin . Değerler eşitse, test başarısız olur. |
3 | assertTrue(expr, msg = None) İfadenin doğru olduğunu test edin . Yanlışsa, test başarısız olur |
4 | assertFalse(expr, msg = None) İfadenin yanlış olduğunu test edin . Doğruysa, test başarısız olur |
5 | assertIs(arg1, arg2, msg = None) Arg1 ve arg2'nin aynı nesneyi değerlendirip değerlendirmediğini test edin . |
6 | assertIsNot(arg1, arg2, msg = None) Arg1 ve arg2'nin aynı nesneyi değerlendirmediğini test edin . |
7 | assertIsNone(expr, msg = None) İfadenin Yok olduğunu test edin . Yok değilse, test başarısız olur |
8 | assertIsNotNone(expr, msg = None) İfadenin Yok olmadığını test edin . Yok ise, test başarısız olur |
9 | assertIn(arg1, arg2, msg = None) Testi o arg1 içindedir ARG2 . |
10 | assertNotIn(arg1, arg2, msg = None) Bu test arg1 değil ARG2 . |
11 | assertIsInstance(obj, cls, msg = None) Testi bu obj bir örneğidir cls'ye |
12 | assertNotIsInstance(obj, cls, msg = None) O Testi obj bir örneği değil cls'ye |
Yukarıdaki iddia işlevlerinden bazıları aşağıdaki kodda uygulanmaktadır -
import unittest
class SimpleTest(unittest.TestCase):
def test1(self):
self.assertEqual(4 + 5,9)
def test2(self):
self.assertNotEqual(5 * 2,10)
def test3(self):
self.assertTrue(4 + 5 == 9,"The result is False")
def test4(self):
self.assertTrue(4 + 5 == 10,"assertion fails")
def test5(self):
self.assertIn(3,[1,2,3])
def test6(self):
self.assertNotIn(3, range(5))
if __name__ == '__main__':
unittest.main()
Yukarıdaki komut dosyası çalıştırıldığında, test2, test4 ve test6 başarısızlık gösterecek ve diğerleri başarıyla çalışacaktır.
FAIL: test2 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "C:\Python27\SimpleTest.py", line 9, in test2
self.assertNotEqual(5*2,10)
AssertionError: 10 == 10
FAIL: test4 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "C:\Python27\SimpleTest.py", line 13, in test4
self.assertTrue(4+5==10,"assertion fails")
AssertionError: assertion fails
FAIL: test6 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "C:\Python27\SimpleTest.py", line 17, in test6
self.assertNotIn(3, range(5))
AssertionError: 3 unexpectedly found in [0, 1, 2, 3, 4]
----------------------------------------------------------------------
Ran 6 tests in 0.001s
FAILED (failures = 3)
İkinci onaylama işlevi kümesi: comparative asserts −
assertAlmostEqual (birinci, ikinci, yerler = 7, msg = Yok, delta = Yok)
Deney bu birinci ve ikinci yaklaşık (ya da yaklaşık olarak) olan ondalık verili bir yuvarlama farkı hesaplayarak eşit basamağa (varsayılan 7),
assertNotAlmostEqual (birinci, ikinci, yerler, mesaj, delta)
Farkı hesaplayarak, verilen ondalık basamak sayısına (varsayılan 7) yuvarlayarak ve sıfırla karşılaştırarak birinci ve ikinci saniyenin yaklaşık olarak eşit olmadığını test edin.
Yukarıdaki işlevlerin her ikisinde de, yerler yerine delta sağlanırsa, birinci ve ikinci arasındaki fark delta'ya eşit veya ondan küçük (veya ondan büyük) olmalıdır.
Hem delta hem de yerleri sağlamak bir TypeError hatası oluşturur.
assertGreater (birinci, ikinci, msg = Yok)
Yöntem adına bağlı olarak ilkinin saniyeden büyük olduğunu test edin . Aksi takdirde test başarısız olur.
assertGreaterEqual (birinci, ikinci, msg = Yok)
Yöntem adına bağlı olarak ilkinin saniyeden büyük veya saniyeye eşit olduğunu test edin . Değilse, test başarısız olur
assertLess (birinci, ikinci, msg = Yok)
Yöntem adına bağlı olarak ilkinin saniyeden küçük olduğunu test edin . Değilse, test başarısız olur
assertLessEqual (birinci, ikinci, msg = Yok)
Yöntem adına bağlı olarak ilkinin saniyeden küçük veya saniyeye eşit olduğunu test edin . Aksi takdirde test başarısız olur.
assertRegexpMatches (metin, regexp, msg = Yok)
Bir regexp aramasının metinle eşleşip eşleşmediğini test edin. Başarısızlık durumunda, hata mesajı modeli ve metni içerecektir. regexp, normal bir ifade nesnesi veya kullanım için uygun bir normal ifade içeren bir dize olabilir.re.search().
assertNotRegexpMatches (metin, regexp, msg = Yok)
Bir regexp aramasının metinle eşleşmediğini doğrular . Model ve eşleşen metin bölümünü içeren bir hata mesajıyla başarısız olur . regexp , normal bir ifade nesnesi veya kullanım için uygun bir normal ifade içeren bir dize olabilir.re.search().
Onaylama işlevleri aşağıdaki örnekte uygulanmaktadır -
import unittest
import math
import re
class SimpleTest(unittest.TestCase):
def test1(self):
self.assertAlmostEqual(22.0/7,3.14)
def test2(self):
self.assertNotAlmostEqual(10.0/3,3)
def test3(self):
self.assertGreater(math.pi,3)
def test4(self):
self.assertNotRegexpMatches("Tutorials Point (I) Private Limited","Point")
if __name__ == '__main__':
unittest.main()
Yukarıdaki komut dosyası test1 ve test4'ü Hata olarak bildirir. 1. testte 22/7 bölümü, 3.14'ün 7 ondalık basamağı içinde değildir. Benzer şekilde, ikinci bağımsız değişken ilk bağımsız değişkendeki metinle eşleştiğinden, test4 AssertionError ile sonuçlanır.
=====================================================FAIL: test1 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "asserttest.py", line 7, in test1
self.assertAlmostEqual(22.0/7,3.14)
AssertionError: 3.142857142857143 != 3.14 within 7 places
================================================================
FAIL: test4 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "asserttest.py", line 13, in test4
self.assertNotRegexpMatches("Tutorials Point (I) Private Limited","Point")
AssertionError: Regexp matched: 'Point' matches 'Point' in 'Tutorials Point (I)
Private Limited'
----------------------------------------------------------------------
Ran 4 tests in 0.001s
FAILED (failures = 2)
Koleksiyonlar için Hak Talebi
Bu onaylama işlevleri kümesi, Python'daki Liste, Tuple, Sözlük ve Küme gibi veri toplama türleriyle kullanılmak üzere tasarlanmıştır.
Sr.No | Yöntem ve Açıklama |
---|---|
1 | assertListEqual (list1, list2, msg = None) İki listenin eşit olduğunu test eder. Değilse, yalnızca ikisi arasındaki farkları gösteren bir hata mesajı oluşturulur. |
2 | assertTupleEqual (tuple1, tuple2, msg = None) İki demetin eşit olduğunu test eder. Değilse, yalnızca ikisi arasındaki farkları gösteren bir hata mesajı oluşturulur. |
3 | assertSetEqual (set1, set2, msg = None) İki setin eşit olduğunu test eder. Değilse, setler arasındaki farkları listeleyen bir hata mesajı oluşturulur. |
4 | assertDictEqual (expected, actual, msg = None) İki sözlüğün eşit olduğunu test edin. Değilse, sözlüklerdeki farklılıkları gösteren bir hata mesajı oluşturulur. |
Aşağıdaki örnek, yukarıdaki yöntemleri uygular -
import unittest
class SimpleTest(unittest.TestCase):
def test1(self):
self.assertListEqual([2,3,4], [1,2,3,4,5])
def test2(self):
self.assertTupleEqual((1*2,2*2,3*2), (2,4,6))
def test3(self):
self.assertDictEqual({1:11,2:22},{3:33,2:22,1:11})
if __name__ == '__main__':
unittest.main()
Yukarıdaki örnekte, test1 ve test3 AssertionError'ı gösterir. Hata mesajı Liste ve Sözlük nesnelerindeki farklılıkları görüntüler.
FAIL: test1 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "asserttest.py", line 5, in test1
self.assertListEqual([2,3,4], [1,2,3,4,5])
AssertionError: Lists differ: [2, 3, 4] != [1, 2, 3, 4, 5]
First differing element 0:
2
1
Second list contains 2 additional elements.
First extra element 3:
4
- [2, 3, 4]
+ [1, 2, 3, 4, 5]
? +++ +++
FAIL: test3 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "asserttest.py", line 9, in test3
self.assertDictEqual({1:11,2:22},{3:33,2:22,1:11})
AssertionError: {1: 11, 2: 22} != {1: 11, 2: 22, 3: 33}
- {1: 11, 2: 22}
+ {1: 11, 2: 22, 3: 33}
? +++++++
----------------------------------------------------------------------
Ran 3 tests in 0.001s
FAILED (failures = 2)
TestLoader sınıfının bir explor () işlevi vardır. Python test çerçevesi bunu basit test keşfi için kullanır. Uyumlu olması için, testleri içeren modüller ve paketler üst düzey dizinden içe aktarılabilir olmalıdır.
Aşağıda, test keşfinin temel komut satırı kullanımı verilmiştir -
Python –m unittest discover
Yorumlayıcı, test içeren tüm modülleri geçerli dizinden ve iç dizinlerden özyinelemeli olarak yüklemeye çalışır. Diğer komut satırı seçenekleri -
Sr.No | Seçenekler ve Açıklama |
---|---|
1 | -v, --verbose Ayrıntılı çıktı |
2 | -s, --start-directory dizin Keşfi başlatmak için dizin (. varsayılan) |
3 | -p, --pattern Test dosyalarını eşleştirmek için desen kalıbı (test * .py varsayılanı) |
4 | -t, --top-level-directory dizin Projenin üst düzey dizini (varsayılan olarak başlangıç dizini) |
Örneğin, isimleri 'testler' dizininde 'assert' ile başlayan modüllerdeki testleri keşfetmek için aşağıdaki komut satırı kullanılır -
C:\python27>python –m unittest –v –s "c:\test" –p "assert*.py"
Test keşif, testleri içe aktararak yükler. Test keşfi, belirttiğiniz başlangıç dizinindeki tüm test dosyalarını bulduğunda, yolları içe aktarılacak paket adlarına dönüştürür.
Başlangıç dizinini bir dizine giden bir yol yerine bir paket adı olarak sağlarsanız, keşfet, hangi konumdan içeri aktarırsa o konumun istediğiniz konum olduğunu varsayar, böylece uyarıyı almazsınız.
Python 2.7'den beri testleri atlama desteği eklenmiştir. Tek tek test yöntemini veya TestCase sınıfını koşullu veya koşulsuz olarak atlamak mümkündür. Çerçeve, belirli bir testin 'beklenen başarısızlık' olarak işaretlenmesine izin verir. Bu test 'başarısız olacak' ancak TestResult'ta başarısız olarak sayılmayacaktır.
Bir yöntemi koşulsuz olarak atlamak için aşağıdaki unittest.skip () sınıfı yöntemi kullanılabilir -
import unittest
def add(x,y):
return x+y
class SimpleTest(unittest.TestCase):
@unittest.skip("demonstrating skipping")
def testadd1(self):
self.assertEquals(add(4,5),9)
if __name__ == '__main__':
unittest.main()
Skip () bir sınıf yöntemi olduğundan, @ belirteci ile ön eklenmiştir. Yöntem bir argüman alır: atlama nedenini açıklayan bir günlük mesajı.
Yukarıdaki komut dosyası çalıştırıldığında, konsolda aşağıdaki sonuç görüntülenir -
C:\Python27>python skiptest.py
s
----------------------------------------------------------------------
Ran 1 test in 0.000s
OK (skipped = 1)
'S' karakteri bir testin atlandığını gösterir.
Testi atlamak için alternatif sözdizimi, test işlevi içinde skipTest () örnek yöntemini kullanmaktır.
def testadd2(self):
self.skipTest("another method for skipping")
self.assertTrue(add(4 + 5) == 10)
Aşağıdaki dekoratörler, test atlama ve beklenen hataları uygular -
Sr.No | Yöntem ve Açıklama |
---|---|
1 | unittest.skip(reason) Koşulsuz olarak dekore edilmiş testi atlayın. nedeni testi atlanır neden kullanılamayacağı açıklamalıdır. |
2 | unittest.skipIf(condition, reason) Koşul doğruysa dekore edilmiş testi atlayın. |
3 | unittest.skipUnless(condition, reason) Koşul doğru olmadığı sürece dekore edilmiş testi atlayın. |
4 | unittest.expectedFailure() Testi beklenen bir başarısızlık olarak işaretleyin. Test çalıştırıldığında başarısız olursa, test başarısız sayılmaz. |
Aşağıdaki örnek, koşullu atlama ve beklenen başarısızlığın kullanımını gösterir.
import unittest
class suiteTest(unittest.TestCase):
a = 50
b = 40
def testadd(self):
"""Add"""
result = self.a+self.b
self.assertEqual(result,100)
@unittest.skipIf(a>b, "Skip over this routine")
def testsub(self):
"""sub"""
result = self.a-self.b
self.assertTrue(result == -10)
@unittest.skipUnless(b == 0, "Skip over this routine")
def testdiv(self):
"""div"""
result = self.a/self.b
self.assertTrue(result == 1)
@unittest.expectedFailure
def testmul(self):
"""mul"""
result = self.a*self.b
self.assertEqual(result == 0)
if __name__ == '__main__':
unittest.main()
Yukarıdaki örnekte, testub () ve testdiv () atlanacaktır. İlk durumda a> b doğrudur, ikinci durumda b == 0 doğru değildir. Öte yandan, testmul () beklenen hata olarak işaretlendi.
Yukarıdaki komut dosyası çalıştırıldığında, atlanan iki test 's' gösterir ve beklenen hata 'x' olarak gösterilir.
C:\Python27>python skiptest.py
Fsxs
================================================================
FAIL: testadd (__main__.suiteTest)
Add
----------------------------------------------------------------------
Traceback (most recent call last):
File "skiptest.py", line 9, in testadd
self.assertEqual(result,100)
AssertionError: 90 != 100
----------------------------------------------------------------------
Ran 4 tests in 0.000s
FAILED (failures = 1, skipped = 2, expected failures = 1)
Python test çerçevesi, istisnaların ortaya çıkıp çıkmadığını kontrol etmek için aşağıdaki onaylama yöntemlerini sağlar.
assertRaises (istisna, çağrılabilir, * değiştirgeler, ** kwds)
Konumsal veya anahtar sözcük bağımsız değişkenleriyle bir işlev çağrıldığında bir istisnanın (ilk bağımsız değişken) ortaya çıktığını test edin. Beklenen istisna ortaya çıkarsa test başarılı olur, başka bir istisna ortaya çıkarsa bir hatadır veya hiçbir istisna ortaya çıkmazsa başarısız olur. Bir istisna grubunu yakalamak için istisna sınıflarını içeren bir demet istisna olarak aktarılabilir.
Aşağıdaki örnekte, ZeroDivisionError'ın yükseltilip yükseltilmediğini kontrol etmek için bir test işlevi tanımlanmıştır.
import unittest
def div(a,b):
return a/b
class raiseTest(unittest.TestCase):
def testraise(self):
self.assertRaises(ZeroDivisionError, div, 1,0)
if __name__ == '__main__':
unittest.main()
Testraise () işlevi, div () işlevi çağrıldığında sıfıra bölmenin gerçekleşip gerçekleşmediğini görmek için assertRaises () işlevini kullanır. Yukarıdaki kod bir istisna oluşturacaktır. Ancak argümanları div () işlevine aşağıdaki gibi değiştirir -
self.assertRaises(ZeroDivisionError, div, 1,1)
Bu değişikliklerle bir kod çalıştırıldığında, ZeroDivisionError oluşmadığından test başarısız olur.
F
================================================================
FAIL: testraise (__main__.raiseTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "raisetest.py", line 7, in testraise
self.assertRaises(ZeroDivisionError, div, 1,1)
AssertionError: ZeroDivisionError not raised
----------------------------------------------------------------------
Ran 1 test in 0.000s
FAILED (failures = 1)
assertRaisesRegexp (istisna, regexp, çağrılabilir, * değiştirgeler, ** kwds)
Normal ifadenin , ortaya çıkarılan istisnanın dize gösterimi üzerinde eşleştiği testler . regexp, normal bir ifade nesnesi veya re.search () tarafından kullanılmaya uygun bir normal ifade içeren bir dize olabilir.
Aşağıdaki örnek, assertRaisesRegexp () öğesinin nasıl kullanıldığını gösterir -
import unittest
import re
class raiseTest(unittest.TestCase):
def testraiseRegex(self):
self.assertRaisesRegexp(TypeError, "invalid", reg,"Point","TutorialsPoint")
if __name__ == '__main__':
unittest.main()
Burada testraseRegex () testi ilk argüman olarak başarısız olmaz. İkinci bağımsız değişken dizesinde "Nokta" bulunur.
================================================================
FAIL: testraiseRegex (__main__.raiseTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "C:/Python27/raiseTest.py", line 11, in testraiseRegex
self.assertRaisesRegexp(TypeError, "invalid", reg,"Point","TutorialsPoint")
AssertionError: TypeError not raised
----------------------------------------------------------------------
Ancak, değişiklik aşağıda gösterildiği gibidir -
self.assertRaisesRegexp(TypeError, "invalid", reg,123,"TutorialsPoint")
TypeError istisnası atılacak. Bu nedenle, aşağıdaki sonuç görüntülenecektir -
================================================================
FAIL: testraiseRegex (__main__.raiseTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "raisetest.py", line 11, in testraiseRegex
self.assertRaisesRegexp(TypeError, "invalid", reg,123,"TutorialsPoint")
AssertionError: "invalid" does not match
"first argument must be string or compiled pattern"
----------------------------------------------------------------------
Java birim test çerçevesi olan Junit (Pyunit, JUnit'in uygulamasıdır) kullanışlı bir zaman aşımı seçeneğine sahiptir. Bir test, belirtilen süreden daha uzun sürerse, başarısız olarak işaretlenecektir.
Python'un test çerçevesi, zaman aşımı için herhangi bir destek içermiyor. Bununla birlikte, zaman aşımı dekoratörü adı verilen üçüncü bir parça modülü işi yapabilir.
Modülü şuradan indirin ve kurun -
https://pypi.python.org/packages/source/t/timeout-decorator/timeout-decorator-0.3.2.tar.gz
- Timeout_decorator'u kodda içe aktar
- Testten önce zaman aşımı dekoratörü koyun
- @timeout_decorator.timeout(10)
Bu çizginin altındaki bir test yöntemi burada belirtilen zaman aşımından (10 dakika) daha uzun sürerse, bir TimeOutError hatası ortaya çıkar. Örneğin -
import time
import timeout_decorator
class timeoutTest(unittest.TestCase):
@timeout_decorator.timeout(5)
def testtimeout(self):
print "Start"
for i in range(1,10):
time.sleep(1)
print "%d seconds have passed" % i
if __name__ == '__main__':
unittest.main()
unittest2, Python 2.7 ve sonrasında Python test çerçevesine eklenen ek özelliklerin bir arka portudur. Python 2.6, 2.7 ve 3. * üzerinde çalışacak şekilde test edilmiştir. En son sürüm şuradan indirilebilir:https://pypi.python.org/pypi/unittest2
Unittest2 yerine unittest2 kullanmak için, import unittest yerine import unittest2 yazmanız yeterlidir.
Unittest2'deki sınıflar, unittest'teki uygun sınıflardan türetilir, bu nedenle, tüm testlerinizi hemen unittest2 kullanmaya geçmek zorunda kalmadan unittest2 test çalıştırma altyapısını kullanmak mümkün olmalıdır. Yeni özellikler uygulamayı planlıyorsanız, test durumunuzuunittest2.TestCase unittest.TestCase yerine
Aşağıdakiler unittest2'nin yeni özellikleridir -
addCleanups daha iyi kaynak yönetimi için
Birçok yeni iddia yöntemi içerir
assertRaises bağlam yöneticisi olarak, daha sonra istisnaya erişim
Modül seviyesi armatürlerine sahiptir. setUpModule ve tearDownModule
İçerir load_tests modüllerden veya paketlerden testleri yüklemek için protokol
startTestRun ve stopTestRun TestResult yöntemleri
Python 2.7'de, unittest komut satırı özelliklerini (test keşfi dahil) şu şekilde çağırırsınız: python -m unittest <args>.
Bunun yerine, unittest2 bir betik birimi2 ile birlikte gelir.
unit2 discover
unit2 -v test_module
Bir test çalıştırması sırasında kontrol-C'nin daha verimli kullanılması, unittest için -c / - catch komut satırı seçeneği ile birlikte sağlanır. catchbreakparametre. Yakalama kopma davranışı etkinleştirildiğinde, control-C o anda çalışan testin tamamlanmasına izin verecek ve test çalıştırması daha sonra sona erecek ve şimdiye kadarki tüm sonuçları raporlayacaktır. İkinci bir kontrol-c, bir KeyboardInterrupt'ı olağan şekilde yükseltecektir.
Birim test işleyicisi çağrılırsa ancak signal.SIGINT işleyicisi kurulu değilse, varsayılan işleyiciyi çağırır. Bu, normalde yüklü bir işleyicinin yerini alan ve ona yetki veren kod tarafından beklenen davranış olacaktır. Birim testi kontrol-c işlemesinin devre dışı bırakılması gereken bağımsız testler için, removeHandler () dekoratörü kullanılabilir.
Aşağıdaki yardımcı program işlevleri, test çerçevelerinde kontrol-c işleme işlevini etkinleştirir -
unittest.installHandler ()
Control-c işleyicisini kurun. Zamansignal.SIGINT alınan tüm kayıtlı sonuçlara TestResult.stop () denir.
unittest.registerResult (sonuç)
Kayıt ol TestResultkontrol-c işleme nesnesi. Bir sonucun kaydedilmesi ona zayıf bir başvuru saklar, bu nedenle sonucun gereksiz yere toplanmasını engellemez.
unittest.removeResult (sonuç)
Kayıtlı bir sonucu kaldırın. Bir sonuç kaldırıldıktan sonra TestResult.stop () artık bir kontrol-c'ye yanıt olarak o sonuç nesnesinde çağrılmayacaktır.
unittest.removeHandler (fonksiyon = Yok)
Bağımsız değişken olmadan çağrıldığında, bu işlev yüklenmişse control-c işleyicisini kaldırır. Bu işlev aynı zamanda test yürütülürken işleyiciyi geçici olarak kaldırmak için bir test dekoratörü olarak da kullanılabilir.
GUI Test Çalıştırıcısı
Unittest modülü, testleri etkileşimli olarak keşfetmek ve çalıştırmak için kurulur. Bu yardımcı program, bir Python betiği 'inittestgui.py', TK grafik araç kiti için bir Python bağlantı noktası olan Tkinter modülünü kullanır. Keşif ve çalıştırma testleri için kullanımı kolay bir GUI sağlar.
Python unittestgui.py
'Testleri Keşfedin' düğmesini tıklayın. Testin çalıştırılacağı dizin ve modülleri seçebileceğiniz küçük bir iletişim kutusu açılır.
Son olarak, başlat düğmesine tıklayın. Testler, seçilen yol ve modül adlarından keşfedilecek ve sonuç bölmesi sonuçları gösterecektir.
Bireysel testin ayrıntılarını görmek için, sonuç kutusunda testi seçin ve tıklayın -
Bu yardımcı programı Python kurulumunda bulamazsanız, proje sayfasından temin edebilirsiniz. http://pyunit.sourceforge.net/.
Benzer şekilde, wxpython araç setine dayalı yardımcı program da burada mevcuttur.
Python 'standart dağıtımı' Doctest 'modülünü içerir. Bu modülün işlevselliği, etkileşimli Python oturumları gibi görünen metin parçalarını aramayı mümkün kılar ve tam olarak gösterildiği gibi çalışıp çalışmadıklarını görmek için bu oturumları yürütür.
Doctest, aşağıdaki senaryolarda çok faydalı olabilir -
Tüm etkileşimli örneklerin hala belgelendiği gibi çalıştığını doğrulayarak bir modülün belge dizilerinin güncel olup olmadığını kontrol etmek için.
Bir test dosyasından veya bir test nesnesinden etkileşimli örneklerin beklendiği gibi çalıştığını doğrulayarak regresyon testi gerçekleştirmek için.
Bir paket için, girdi-çıktı örnekleriyle serbestçe gösterilen öğretici dokümantasyon yazmak için
Python'da bir 'docstring', bir sınıf, işlev veya modülde ilk ifade olarak görünen bir dize değişmezidir. Paket yürütüldüğünde yok sayılır, ancak derleyici tarafından tanınır ve__doc__çevreleyen sınıf, işlev veya modülün niteliği. İç gözlem yoluyla kullanılabildiğinden, nesnenin dokümantasyonu için kanonik yerdir.
Docstring'in içine Python kodunun farklı bölümlerinin örnek kullanımını koymak olağan bir uygulamadır. Doctest modülü, bu belge dizilerinin koddaki aralıklı revizyonlarla güncel olduğunu doğrulamaya izin verir.
Aşağıdaki kodda, faktöriyel bir fonksiyon örnek kullanımla serpiştirilmiş olarak tanımlanmıştır. Örnek kullanımın doğru olup olmadığını doğrulamak için doctest modülündeki testmod () işlevini çağırın.
"""
This is the "example" module.
The example module supplies one function, factorial(). For example,
>>> factorial(5)
120
"""
def factorial(x):
"""Return the factorial of n, an exact integer >= 0.
>>> factorial(-1)
Traceback (most recent call last):
...
ValueError: x must be >= 0
"""
if not x >= 0:
raise ValueError("x must be >= 0")
f = 1
for i in range(1,x+1):
f = f*i
return f
if __name__ == "__main__":
import doctest
doctest.testmod()
Yukarıdaki komut dosyasını FactDocTest.py olarak girin ve kaydedin ve bu komut dosyasını komut satırından çalıştırmayı deneyin.
Python FactDocTest.py
Örnek başarısız olmadıkça hiçbir çıktı gösterilmeyecektir. Şimdi, komut satırını şu şekilde değiştirin -
Python FactDocTest.py –v
Konsol şimdi aşağıdaki çıktıyı gösterecek -
C:\Python27>python FactDocTest.py -v
Trying:
factorial(5)
Expecting:
120
ok
Trying:
factorial(-1)
Expecting:
Traceback (most recent call last):
...
ValueError: x must be >= 0
ok
2 items passed all tests:
1 tests in __main__
1 tests in __main__.factorial
2 tests in 2 items.
2 passed and 0 failed.
Test passed.
Öte yandan, factorial () işlevinin kodu docstring'de beklenen sonucu vermezse, başarısızlık sonucu görüntülenir. Örneğin, yukarıdaki komut dosyasında f = 1 yerine f = 2'yi değiştirin ve doctest'i yeniden çalıştırın. Sonuç aşağıdaki gibi olacaktır -
Trying:
factorial(5)
Expecting:
120
**********************************************************************
File "docfacttest.py", line 6, in __main__
Failed example:
factorial(5)
Expected:
120
Got:
240
Trying:
factorial(-1)
Expecting:
Traceback (most recent call last):
...
ValueError: x must be >= 0
ok
1 items passed all tests:
1 tests in __main__.factorial
**********************************************************************
1 items had failures:
1 of 1 in __main__
2 tests in 2 items.
1 passed and 1 failed.
***Test Failed*** 1 failures.
Doctest: Bir Metin Dosyasındaki Örnekleri Kontrol Etme
Doctest'in diğer bir basit uygulaması, bir metin dosyasında etkileşimli örnekleri test etmektir. Bu, testfile () işlevi ile yapılabilir.
Aşağıdaki metin, "example.txt" adlı bir metin dosyasında saklanır.
Using ''factorial''
-------------------
This is an example text file in reStructuredText format. First import
''factorial'' from the ''example'' module:
>>> from example import factorial
Now use it:
>>> factorial(5)
120
Dosya içeriği docstring olarak değerlendirilir. Metin dosyasındaki örnekleri doğrulamak için doctest modülünün testfile () işlevini kullanın.
def factorial(x):
if not x >= 0:
raise ValueError("x must be >= 0")
f = 1
for i in range(1,x+1):
f = f*i
return f
if __name__ == "__main__":
import doctest
doctest.testfile("example.txt")
Testmod () ile olduğu gibi, testfile () bir örnek başarısız olmadıkça hiçbir şey göstermez. Bir örnek başarısız olursa, başarısız olan örnekler ve hataların nedenleri, testmod () ile aynı format kullanılarak konsola yazdırılır.
Çoğu durumda, etkileşimli bir konsol oturumunun kopyalayıp yapıştırması iyi çalışır, ancak doctest herhangi bir belirli Python kabuğunun tam bir öykünmesini yapmaya çalışmaz.
Beklenen herhangi bir çıktı, kodu içeren son '>>>' veya '...' satırını hemen takip etmelidir ve beklenen çıktı (varsa) bir sonraki '>>>' veya tamamen boşluk satırına uzanır.
Beklenen çıktı tamamen boşluklu bir satır içeremez, çünkü böyle bir satır beklenen çıktının sonunu işaret etmek için alınır. Beklenen çıktı boş bir satır içeriyorsa, doctest örneğinizde boş bir satır olması beklenen her yere <BLANKLINE> yazın.
Doctest API, docstrings'den etkileşimli örnekleri depolamak için kullanılan aşağıdaki iki kapsayıcı sınıfı etrafında döner:
Example - Beklenen çıktı ile eşleştirilmiş tek bir Python ifadesi.
DocTest - Tipik olarak tek bir doküman dizesinden veya bir metin dosyasından çıkarılan bir Örnekler koleksiyonu.
Aşağıdaki ek işleme sınıfları bulmak, ayrıştırmak ve çalıştırmak ve doctest örneklerini kontrol etmek için tanımlanmıştır -
DocTestFinder - Belirli bir modüldeki tüm doküman dizilerini bulur ve etkileşimli örnekler içeren her dokümandan bir DocTest oluşturmak için bir DocTestParser kullanır.
DocTestParser - Bir dizeden (bir nesnenin docstring'i gibi) bir doctest nesnesi oluşturur.
DocTestRunner - Örnekleri bir doctest'te yürütür ve çıktılarını doğrulamak için bir OutputChecker kullanır.
OutputChecker - Bir doctest örneğinden alınan gerçek çıktıyı beklenen çıktıyla karşılaştırır ve eşleşip eşleşmediklerine karar verir.
DocTestFinder Sınıfı
Belli bir nesne ile ilgili olan doctestlerini, onun docstring'inden ve içerdiği nesnelerin docstrings'inden çıkarmak için kullanılan bir işleme sınıfıdır. Belge testleri şu anda aşağıdaki nesne türlerinden (modüller, işlevler, sınıflar, yöntemler, statik yöntemler, sınıf yöntemleri ve özellikler) çıkarılabilir.
Bu sınıf, find () yöntemini tanımlar. Bu tarafından tanımlanan doctestleri bir listesini verir nesnenin docstringe, ya da ihtiva edilen nesnelerin Docstringler herhangi biri ile.
DocTestParser Sınıfı
Bir dizeden etkileşimli örnekleri çıkarmak ve bunları bir DocTest nesnesi oluşturmak için kullanmak için kullanılan bir işleme sınıfıdır. Bu sınıf aşağıdaki yöntemleri tanımlar -
get_doctest() - Verilen dizeden tüm doctest örneklerini çıkarın ve bunları bir DocTest nesne.
get_examples(string[, name]) - Verilen dizeden tüm doctest örneklerini çıkarın ve bunları bir liste olarak geri döndürün. Examplenesneler. Satır numaraları 0 tabanlıdır. İsteğe bağlı bağımsız değişken adı, bu dizeyi tanımlayan bir addır ve yalnızca hata mesajları için kullanılır.
parse(string[, name]) - Verilen dizeyi örneklere ve araya giren metne bölün ve bunları bir alternatif liste olarak geri döndürün Examplesve dizeler. Satır numaralarıExamples0 tabanlıdır. İsteğe bağlı bağımsız değişken adı, bu dizeyi tanımlayan bir addır ve yalnızca hata mesajları için kullanılır.
DocTestRunner Sınıfı
Bu, bir DocTest'teki etkileşimli örnekleri yürütmek ve doğrulamak için kullanılan bir işleme sınıfıdır. Aşağıdaki yöntemler içinde tanımlanmıştır -
report_start ()
Test çalıştırıcısının verilen örneği işlemek üzere olduğunu bildirin. Bu yöntem, alt sınıflara izin vermek için sağlanmıştır.DocTestRunnerçıktılarını özelleştirmek için; doğrudan aranmamalı
report_success ()
Verilen örneğin başarıyla çalıştığını bildirin. Bu yöntem, DocTestRunner alt sınıflarının çıktılarını özelleştirmesine izin vermek için sağlanmıştır; doğrudan aranmamalıdır.
report_failure ()
Verilen örneğin başarısız olduğunu bildirin. Bu yöntem, alt sınıflara izin vermek için sağlanmıştır.DocTestRunnerçıktılarını özelleştirmek için; doğrudan aranmamalıdır.
report_unexpected_exception ()
Verilen örneğin beklenmedik bir istisna oluşturduğunu bildirin. Bu yöntem, DocTestRunner alt sınıflarının çıktılarını özelleştirmesine izin vermek için sağlanmıştır; doğrudan aranmamalıdır.
çalıştır (test)
Örnekler çalıştırın testi (bir doctest nesne) ve yazar fonksiyonu kullanarak sonuçları gösterir out .
özetlemek ([verbose])
Bu DocTestRunner tarafından çalıştırılan tüm test olaylarının bir özetini yazdırın ve adlandırılmış bir tuple TestResults (başarısız, denendi) döndürün. İsteğe bağlı ayrıntılı bağımsız değişken, özetin ne kadar ayrıntılı olduğunu kontrol eder. Ayrıntı belirtilmezse, DocTestRunner'ın ayrıntı düzeyi kullanılır.
OutputChecker Sınıfı
Bu sınıf, bir doctest örneğinin gerçek çıktısının beklenen çıktıyla eşleşip eşleşmediğini kontrol etmek için kullanılır.
Bu sınıfta aşağıdaki yöntemler tanımlanmıştır -
check_output ()
Dönüş TrueBir örnek gerçek çıktı if ( var beklenen çıkışı (ile eşleşmeleri) Talep ). Bu dizeler aynı iseler her zaman eşleşmiş olarak kabul edilirler; ancak test koşucunun kullandığı seçenek işaretlerine bağlı olarak, birkaç tam olmayan eşleme türü de mümkündür. Seçenek bayrakları hakkında daha fazla bilgi için Seçenek Bayrakları ve Yönergeler bölümüne bakın .
çıktı_ farkı ()
Verilen örnekte (beklenen çıkışı arasındaki farkları açıklamayı döndürür örnek ) ve gerçek çıkış ( var ).
Unittest ile DocTest Entegrasyonu
Doctest modülü, modüllerden ve doctestleri içeren metin dosyalarından unittest test takımları oluşturmak için kullanılabilecek iki işlev sağlar. Unittest test keşfi ile entegre etmek için test modülünüze bir load_tests () fonksiyonu ekleyin -
import unittest
import doctest
import doctestexample
def load_tests(loader, tests, ignore):
tests.addTests(doctest.DocTestSuite(doctestexample))
return tests
Doctest yanı sıra unittest'ten testlerin birleşik bir TestSuite'i oluşturulacak ve artık unittest modülünün main () yöntemi veya run () yöntemi ile yürütülebilir.
Aşağıdakiler oluşturmak için iki ana işlevdir unittest.TestSuite doctestleri ile metin dosyalarından ve modüllerden örnekler -
doctest.DocFileSuite ()
Doctest testlerini bir veya daha fazla metin dosyasından bir unittest.TestSuite. Döndürülen unittest.TestSuite, unittest çerçevesi tarafından çalıştırılacak ve her dosyadaki etkileşimli örnekleri çalıştıracaktır. Bir dosyadaki örneklerden herhangi biri başarısız olursa, sentezlenmiş birim testi başarısız olur vefailureException Testi içeren dosyanın adını ve (bazen yaklaşık) bir satır numarasını gösteren bir istisna ortaya çıkar.
doctest.DocTestSuite ()
Bir modül için doctest testlerini bir unittest.TestSuite.
Döndürülen unittest.TestSuite, unittest çerçevesi tarafından çalıştırılacak ve modüldeki her bir doctest'i çalıştıracaktır. Herhangi bir doctest başarısız olursa, sentezlenmiş birim testi başarısız olur vefailureException testi içeren dosyanın adını ve (bazen yaklaşık) bir satır numarasını gösteren bir istisna oluşur
Kapakların altında DocTestSuite () bir unittest.TestSuite doctest.DocTestCase örneklerinin dışında ve DocTestCase, unittest.TestCase öğesinin bir alt sınıfıdır.
Benzer şekilde, DocFileSuite (), doctest.DocFileCase örneklerinden bir unittest.TestSuite oluşturur ve DocFileCase, DocTestCase'in bir alt sınıfıdır.
Yani bir unittest.TestSuite oluşturmanın her iki yolu da DocTestCase'in örneklerini çalıştırır. Doctest işlevlerini kendiniz çalıştırdığınızda, seçenek bayraklarını doctest işlevlerine geçirerek doğrudan kullanımdaki doctest seçeneklerini kontrol edebilirsiniz.
Bununla birlikte, bir unittest çerçevesi yazıyorsanız, unittest nihayetinde testlerin ne zaman ve nasıl çalıştırılacağını kontrol eder. Çerçeve yazarı tipik olarak doctest raporlama seçeneklerini kontrol etmek ister (örneğin, komut satırı seçenekleriyle belirtilmiş olabilir), ancak seçenekleri unittest üzerinden doctest test çalıştırıcılarına geçirmenin bir yolu yoktur.
2004 yılında Holger Krekel kendi adını değiştirdi. stdadı genellikle Python ile birlikte gelen Standart Kitaplığın adı ile karıştırılan paket (sadece biraz daha az kafa karıştırıcı olan) 'py' adını taşır. Paket birkaç alt paket içerse de, artık neredeyse tamamen py.test çerçevesi ile biliniyor.
Py.test çerçevesi, Python testi için yeni bir standart oluşturdu ve bugün birçok geliştirici tarafından çok popüler hale geldi. Test yazımı için sunduğu zarif ve Pythonic deyimler, test süitlerinin çok daha kompakt bir tarzda yazılmasını mümkün kılmıştır.
py.test, Python'un standart unittest modülüne standart olmayan bir alternatiftir. Tam özellikli ve genişletilebilir bir test aracı olmasına rağmen, basit bir sözdizimine sahiptir. Bir test paketi oluşturmak, birkaç işlevi olan bir modül yazmak kadar kolaydır.
py.test, tüm POSIX işletim sistemlerinde ve WINDOWS (XP / 7/8) üzerinde Python sürüm 2.6 ve üzeri ile çalışır.
Kurulum
Pytest modülünü geçerli Python dağıtımına ve ayrıca py.test.exe yardımcı programına yüklemek için aşağıdaki kodu kullanın. Testler her ikisi kullanılarak çalıştırılabilir.
pip install pytest
Kullanım
Test beklentilerini ortaya koymak için assert ifadesini kullanabilirsiniz. pytest'in iddialı içgözlemi, assert ifadesinin ara değerlerini akıllıca rapor ederek sizi birçok ismini öğrenme ihtiyacından kurtaracaktır.JUnit legacy methods.
# content of test_sample.py
def func(x):
return x + 1
def test_answer():
assert func(3) == 5
Yukarıdaki testi çalıştırmak için aşağıdaki komut satırını kullanın. Test çalıştırıldığında, aşağıdaki sonuç konsolda görüntülenir -
C:\Python27>scripts\py.test -v test_sample.py
============================= test session starts =====================
platform win32 -- Python 2.7.9, pytest-2.9.1, py-1.4.31, pluggy-0.3.1 -- C:\Pyth
on27\python.exe
cachedir: .cache
rootdir: C:\Python27, inifile:
collected 1 items
test_sample.py::test_answer FAILED
================================== FAILURES =====================
_________________________________ test_answer _________________________________
def test_answer():
> assert func(3) == 5
E assert 4 == 5
E + where 4 = func(3)
test_sample.py:7: AssertionError
========================== 1 failed in 0.05 seconds ====================
Test, –m anahtarı kullanılarak pytest modülü dahil edilerek komut satırından da çalıştırılabilir.
python -m pytest test_sample.py
Bir Sınıfta Birden Çok Testi Gruplama
Birkaç testten fazlasını yapmaya başladığınızda, testleri mantıksal olarak, sınıflarda ve modüllerde gruplamak genellikle mantıklıdır. İki test içeren bir sınıf yazalım -
class TestClass:
def test_one(self):
x = "this"
assert 'h' in x
def test_two(self):
x = "hello"
assert hasattr(x, 'check')
Aşağıdaki test sonucu görüntülenecektir -
C:\Python27>scripts\py.test -v test_class.py
============================= test session starts =====================
platform win32 -- Python 2.7.9, pytest-2.9.1, py-1.4.31, pluggy-0.3.1 -- C:\Pyt
on27\python.exe
cachedir: .cache
rootdir: C:\Python27, inifile:
collected 2 items
test_class.py::TestClass::test_one PASSED
test_class.py::TestClass::test_two FAILED
================================== FAILURES =====================
_____________________________ TestClass.test_two ______________________________
self = <test_class.TestClass instance at 0x01309DA0>
def test_two(self):
x = "hello"
> assert hasattr(x, 'check')
E assert hasattr('hello', 'check')
test_class.py:7: AssertionError
===================== 1 failed, 1 passed in 0.06 seconds ======================
Burun projesi bir yıl sonra 2005 yılında yayınlandı py.testmodern görünümüne kavuştu. Jason Pellerin tarafından, py.test'in öncülüğünü yaptığı aynı test deyimlerini desteklemek için yazılmıştır, ancak kurulumu ve bakımı daha kolay olan bir pakette.
nose modül pip yardımcı programı yardımıyla kurulabilir
pip install nose
Bu, burun modülünü mevcut Python dağıtımına ve bir nosetest.exe aracına yükleyecektir, bu da testin –m anahtarının yanı sıra bu yardımcı program kullanılarak çalıştırılabileceği anlamına gelir.
C:\python>nosetests –v test_sample.py
Or
C:\python>python –m nose test_sample.py
nose test toplar unittest.TestCaseelbette alt sınıflar. Unittest.TestCase'in alt sınıfları olmayan test sınıflarının yanı sıra basit test fonksiyonları da yazabiliriz. Burun ayrıca zamanlı testler yazmak, istisnaları test etmek ve diğer yaygın kullanım durumları için bir dizi yararlı işlev sunar.
nosetestleri otomatik olarak toplar. Test senaryolarını test paketlerine manuel olarak toplamanıza gerek yoktur. Testleri çalıştırmak duyarlıdır, çünkünose ilk test modülü yüklenir yüklenmez testleri çalıştırmaya başlar.
Birim test modülünde olduğu gibi, nose paket, modül, sınıf ve test senaryosu düzeyinde armatürleri destekler, böylece pahalı başlatma mümkün olduğunca seyrek yapılabilir.
Temel Kullanım
Nosetest.py'yi daha önce kullanılan betiğe benzer düşünelim -
# content of nosetest.py
def func(x):
return x + 1
def test_answer():
assert func(3) == 5
Yukarıdaki testi çalıştırmak için aşağıdaki komut satırı sözdizimini kullanın -
C:\python>nosetests –v nosetest.py
Konsolda görüntülenen çıktı aşağıdaki gibi olacaktır -
nosetest.test_answer ... FAIL
================================================================
FAIL: nosetest.test_answer
----------------------------------------------------------------------
Traceback (most recent call last):
File "C:\Python34\lib\site-packages\nose\case.py", line 198, in runTest
self.test(*self.arg)
File "C:\Python34\nosetest.py", line 6, in test_answer
assert func(3) == 5
AssertionError
----------------------------------------------------------------------
Ran 1 test in 0.000s
FAILED (failures = 1)
nose kullanılarak DocTest ile entegre edilebilir with-doctest athe bove komut satırındaki seçenek.
\nosetests --with-doctest -v nosetest.py
Kullanabilirsin nose bir test komut dosyasında -
import nose
nose.main()
Test komut dosyasının başarı durumunda 0 ve başarısızlık durumunda 1 ile çıkmasını istemiyorsanız (unittest.main gibi), bunun yerine nose.run () kullanın -
import nose
result = nose.run()
Test çalıştırması başarılı olursa sonuç doğru, başarısız olursa veya yakalanmamış bir istisna ortaya çıkarsa yanlış olacaktır.
nosepaket, modül, sınıf ve test düzeyinde armatürleri (kurulum ve sökme yöntemleri) destekler. Py.test veya unittest fikstürlerinde olduğu gibi, kurulum her zaman herhangi bir testten (veya test paketleri ve modülleri için testler toplamasından) önce çalışır; Test çalıştırmasının durumuna bakılmaksızın kurulum başarıyla tamamlanırsa sökme işlemi çalışır.
Burun.tools modülü, test yürütme süresini kısıtlamak ve istisnaları test etmek için dekoratörler ve unittest.TestCase'de bulunan aynı assertX yöntemlerinin tümü dahil, yararlı bulabileceğiniz bir dizi test yardımcısı sağlar.
nose.tools.ok_(expr, msg = None) - İddia etmenin kısaltması.
nose.tools.eq_(a, b, msg = None) - "a == b" için kısa açıklama, "% r! =% R"% (a, b)
nose.tools.make_decorator(func) - Burun ek öğeleri (yani, kurulum ve sökme) dahil olmak üzere dekore edilmiş işlevin meta verilerini düzgün şekilde çoğaltmak için bir test dekoratörünü sarar.
nose.tools.raises(*exceptions) - Testin geçmek için beklenen istisnalardan birini ortaya çıkarması gerekir.
nose.tools.timed(limit) - Testin geçmesi için belirtilen süre içinde bitmesi gerekir
nose.tools.istest(func) - Dekoratör bir işlevi veya yöntemi test olarak işaretlemek için
nose.tools.nottest(func) - Dekoratörün bir işlevi veya yöntemi test değil olarak işaretlemesi
Parametreli Test
Python'un test çerçevesi olan unittest, parametreleştirilmiş test durumlarını çalıştırmanın basit bir yoluna sahip değildir. Başka bir deyişle, argümanları kolayca birunittest.TestCase dışarıdan.
Bununla birlikte, pytest modül portları parametrelendirmeyi birkaç iyi entegre yolla test eder -
pytest.fixture() fikstür fonksiyonları seviyesinde parametrelendirme tanımlamanıza izin verir.
@pytest.mark.parametrizefonksiyon veya sınıf düzeyinde parametrelendirmeyi tanımlamaya izin verir. Belirli bir test fonksiyonu veya sınıfı için birden fazla argüman / fikstür seti sağlar.
pytest_generate_tests kendi özel dinamik parametrelendirme şemanızı veya uzantılarınızı uygulamanızı sağlar.
'Burun parametreli' üçüncü taraf modülü, herhangi bir Python test çerçevesi ile Parametreli teste izin verir. Bu bağlantıdan indirilebilir -https://github.com/wolever/nose-parameterized