From add1926ec39fa0fae22ecd7aa88475b5693db881 Mon Sep 17 00:00:00 2001 From: Kyber-strannik Date: Mon, 17 Nov 2025 13:42:21 +0300 Subject: [PATCH 1/2] Diplom_unit_for_burgers --- README.md | 24 --------- Requirements.txt | 2 + conftest.py | 26 +++++++++ __init__.py => praktikum/__init__.py | 0 .../__pycache__/__init__.cpython-313.pyc | Bin 0 -> 153 bytes praktikum/__pycache__/bun.cpython-313.pyc | Bin 0 -> 1078 bytes praktikum/__pycache__/burger.cpython-313.pyc | Bin 0 -> 3238 bytes .../__pycache__/database.cpython-313.pyc | Bin 0 -> 2447 bytes .../__pycache__/ingredient.cpython-313.pyc | Bin 0 -> 1402 bytes .../ingredient_types.cpython-313.pyc | Bin 0 -> 383 bytes bun.py => praktikum/bun.py | 0 burger.py => praktikum/burger.py | 0 database.py => praktikum/database.py | 0 ingredient.py => praktikum/ingredient.py | 0 .../ingredient_types.py | 0 praktikum.py => praktikum/praktikum.py | 0 tests/__init__.py | 0 tests/__pycache__/__init__.cpython-313.pyc | Bin 0 -> 151 bytes .../test_bun.cpython-313-pytest-8.4.1.pyc | Bin 0 -> 3165 bytes .../test_burger.cpython-313-pytest-8.4.1.pyc | Bin 0 -> 10562 bytes ...test_database.cpython-313-pytest-8.4.1.pyc | Bin 0 -> 5110 bytes ...st_ingredient.cpython-313-pytest-8.4.1.pyc | Bin 0 -> 6186 bytes tests/test_bun.py | 17 ++++++ tests/test_burger.py | 50 ++++++++++++++++++ tests/test_database.py | 30 +++++++++++ tests/test_ingredient.py | 50 ++++++++++++++++++ 26 files changed, 175 insertions(+), 24 deletions(-) create mode 100644 Requirements.txt create mode 100644 conftest.py rename __init__.py => praktikum/__init__.py (100%) create mode 100644 praktikum/__pycache__/__init__.cpython-313.pyc create mode 100644 praktikum/__pycache__/bun.cpython-313.pyc create mode 100644 praktikum/__pycache__/burger.cpython-313.pyc create mode 100644 praktikum/__pycache__/database.cpython-313.pyc create mode 100644 praktikum/__pycache__/ingredient.cpython-313.pyc create mode 100644 praktikum/__pycache__/ingredient_types.cpython-313.pyc rename bun.py => praktikum/bun.py (100%) rename burger.py => praktikum/burger.py (100%) rename database.py => praktikum/database.py (100%) rename ingredient.py => praktikum/ingredient.py (100%) rename ingredient_types.py => praktikum/ingredient_types.py (100%) rename praktikum.py => praktikum/praktikum.py (100%) create mode 100644 tests/__init__.py create mode 100644 tests/__pycache__/__init__.cpython-313.pyc create mode 100644 tests/__pycache__/test_bun.cpython-313-pytest-8.4.1.pyc create mode 100644 tests/__pycache__/test_burger.cpython-313-pytest-8.4.1.pyc create mode 100644 tests/__pycache__/test_database.cpython-313-pytest-8.4.1.pyc create mode 100644 tests/__pycache__/test_ingredient.cpython-313-pytest-8.4.1.pyc create mode 100644 tests/test_bun.py create mode 100644 tests/test_burger.py create mode 100644 tests/test_database.py create mode 100644 tests/test_ingredient.py diff --git a/README.md b/README.md index 272081708..e69de29bb 100644 --- a/README.md +++ b/README.md @@ -1,24 +0,0 @@ -## Задание 1: Юнит-тесты - -### Автотесты для проверки программы, которая помогает заказать бургер в Stellar Burgers - -### Реализованные сценарии - -Созданы юнит-тесты, покрывающие классы `Bun`, `Burger`, `Ingredient`, `Database` - -Процент покрытия 100% (отчет: `htmlcov/index.html`) - -### Структура проекта - -- `praktikum` - пакет, содержащий код программы -- `tests` - пакет, содержащий тесты, разделенные по классам. Например, `bun_test.py`, `burger_test.py` и т.д. - -### Запуск автотестов - -**Установка зависимостей** - -> `$ pip install -r requirements.txt` - -**Запуск автотестов и создание HTML-отчета о покрытии** - -> `$ pytest --cov=praktikum --cov-report=html` diff --git a/Requirements.txt b/Requirements.txt new file mode 100644 index 000000000..e19767c79 --- /dev/null +++ b/Requirements.txt @@ -0,0 +1,2 @@ +pytest==8.4.1 +pytest-cov==6.2.1 \ No newline at end of file diff --git a/conftest.py b/conftest.py new file mode 100644 index 000000000..cb6868f2a --- /dev/null +++ b/conftest.py @@ -0,0 +1,26 @@ +import pytest +from unittest.mock import Mock +from praktikum.ingredient_types import INGREDIENT_TYPE_FILLING, INGREDIENT_TYPE_SAUCE + +@pytest.fixture +def mock_bun(): + mock_bun = Mock() + mock_bun.get_name.return_value = "yellow bun" + mock_bun.get_price.return_value = 400 + return mock_bun + +@pytest.fixture +def mock_sauce(): + mock_sauce = Mock() + mock_sauce.get_name.return_value = "teriyaki" + mock_sauce.get_price.return_value = 400 + mock_sauce.get_type.return_value = INGREDIENT_TYPE_SAUCE + return mock_sauce + +@pytest.fixture +def mock_filling(): + mock_filling = Mock() + mock_filling.get_name.return_value = "snake" + mock_filling.get_price.return_value = 400 + mock_filling.get_type.return_value = INGREDIENT_TYPE_FILLING + return mock_filling diff --git a/__init__.py b/praktikum/__init__.py similarity index 100% rename from __init__.py rename to praktikum/__init__.py diff --git a/praktikum/__pycache__/__init__.cpython-313.pyc b/praktikum/__pycache__/__init__.cpython-313.pyc new file mode 100644 index 0000000000000000000000000000000000000000..f9c936e8505cb6fd5f04dc9c301f56b8b4ef5836 GIT binary patch literal 153 zcmey&%ge<81W%=eGePuY5CH>>P{wB#AY&>+I)f&o-%5reCLr%KNa~h_vsFxJacWU< zOkz=KN@7fLUSf7?j7w%gPJS+wiZ_f&DlJL}sfmxz%*!l^kJl@xyv1RYo1apelWJGQ V3N#30Q89?|k(rT^v4|PS0ss_#B@X}q literal 0 HcmV?d00001 diff --git a/praktikum/__pycache__/bun.cpython-313.pyc b/praktikum/__pycache__/bun.cpython-313.pyc new file mode 100644 index 0000000000000000000000000000000000000000..6af0eea6e33361e9bad149c6a816142f500eb525 GIT binary patch literal 1078 zcmbtSzi-n(6h8k-9QPMcBBcfi3s|y50@?*320)A*uqt)1baE3nGPq7T+e124R4P;| zsu&`5;13{M`BA7S>ECePDzAuvEs#oGdG8XJij^C__rCY;zWeUkgNX@)!1`fM`yGjp z?@02cj*>wIk`Aem>%=A|v86eA*kPn8SCx9cALH4X$3Qy7AxyFflWmD9w#-ypaWq?X z3bU%MndGU;igv-#W@Wo5m=3G}Oc$(VmzdEss$-e*APIkhJ(9cvF!)=3#6R(Uewc1s z{C&E?5BX90l7Hm~7Ss>Z9Z^l!`A5DFxK}gyn{&JNE&dG@KJy+w!UKEh3l!?{J^qD% z0POJt4x2!2SOWz41FDrkD*#lLOpUx?`EMmc8OyS=I}h(V3z5en$7M;wb)wLH<~fi2)u6R9 z3iOt@^+P|VG#iHr)^b(^&1f88og81TY?qG9=4)dBoMZ*&#R&`R zXC9%5xp<8UNrxBUqGOnAu|_Hr*%(>v|0 zHrFd`5~}~>0f)&hqVzcjCA?F7tZV|uaYpgBa~{^V*0??+Jg^e9=HU;7Nuqa{q(SLQ ztC0klk5l?Qaf4B3iqd7DMRDMVUf6m$aGP8z%%p zzaVMm?%XNl=1>lm=HU|w#cL>wPal1sd&6hlXBO1NRa21k6-km#O#sA!KdEaT As{jB1 literal 0 HcmV?d00001 diff --git a/praktikum/__pycache__/burger.cpython-313.pyc b/praktikum/__pycache__/burger.cpython-313.pyc new file mode 100644 index 0000000000000000000000000000000000000000..8d3e808a0d8aa7130ea18eef489fbcc052c42c8c GIT binary patch literal 3238 zcmbVO-EY%Y6u-7($4S#BrIdCBT2m_85Gd;kgO$$}*t#+*RI5@28CGabWoVsn?eNhj zJ|-xP3Gsl|hdmT&6638K>&E!#_!Dde7GwztY2tym@*(lMbL=>2!-upTDfgap?mg%J z?(d#^<5!iHVFL23p+@5cLjFO+mq0;zwFQ)GL?$w`hx9U(kx*y%u)Q4RT%Fq!=;bLd zk$TchSb#us@?zP30kAr=U zBnf3?LRp!ioXpaI%ry|o%Yme@o|X9qawI^5vXB(P3c6NE7L#GHLatSTmK+98K(25- z5jm2qgz-w(ssbwtR+VeT{xB=E(@Lzxocfg14`_?T8pMk8m zp}p@${3^A9e;VfD_#JB+C*FnmZq{@5nK-OA`F5!Non+mE6@Q1JS;_j#MK%Mj8T%rn zJBy$p$vIeT8n$o+J&*}E<%QrzU=0ksPzWf1wa>UG!5$}!Z@!v;>3oJ~8s zk^_cHjbw^uhf_&IPaRd0yR^(`daTeX9mxz$9W}M1*|Fpy_Z-_ZryNmHG+i?l#ff@{ z+wMD}klCn)<}`U+Q#TQ~EIwYlZX)nVtbc`2X#|>~LQSKC6!#Sa;~E)^m3B2kDBh|# z<3`0*C%>BkvmGQeyoa4!)bGfOLBmLsGx8C4(lu`dDN0_;_(;bEZCzr9XvfzE|)Hw+ut zhZxC=^|n}_7n^LcX)63kOnBSkHGNn;xnM^{ZW|%?eLEUpLPgY!i`qxKv>a_URmalD z)%k$y3xMAm`3QsYOaXHwmX9p6Bg-ZS?8r(FZAe#7KpsRqpHX-_y)(cncpu_-oxz~= z&eP%{O_`KV%YR!-r^XddP1E68m=2doXX2b!c1Q+og2q6+l3#q$azd6A2r2U~!^l=> z-af)mKC#(OY|baP+llQD_*~+PN8*{tHDLU_Oe>tOZ(h z1_~T=l}QchsfAH?aM8PH_H(R->eiXPb%az9i3~6;B+m9d$9e<4@Qyp75xA8Zstu`L zO?fmryytQ&dOne~4TE9@@`|x8;h)10&=y<#gLqZU$C~X}b1v30bnP4G_}8uJaS zA2zI>=zSci&ez86+IX%uVMh{|cE8}r(!`uVVzm7PIUi|522GQt?-_G2Z3pPy33@5vy(05ki~xdKPi*aEr!wy^^FT{ zTS268LBc*h&Tf?)MjB_Ot>Zz@6&?@n!*c9`fp9$JRjG)Trl-cdBBk}Pio+Qub=(2- zR7Q39(ew$Gdi5DNlGgOXi{J#QI;3hDlfG5VP6ehd3_FYF6?MUf4}mcY4V1OCYRbHE z^7_eq>n6K(Q?9kkTDAH1r&eRPwS4D9a6x(3TmH`)IaH`SYO*=V@oew%5?9cBWw095YcJH@#9k333V=X=ScSP~;LMV2n>SER8lBvKA@w%9o zI&7&UCv{p)>!*9Hx~&hzZI8vMJEd-F@r}mojky&amh_2L*ZELf|1?^2<RFvWL zkfKohtDyKE(Z$%{!%gu&#NjmEq|4D#+Q|9DPh_`1oRuL*Q>0>iKQxdHVxsJBcR zsJEcr!%WF@VDvE4GCF&hIYHoSCpWw%(9Oj{eDkFIhCt^njf?fggN^{Tg#Yt+J)O$p zH&r`)x464@7MFab?*Wen7x^{_EbYb-w|ES1qunropC1Mm?;OW4%o9@gj4Xdb5^sb! M)BF-Xpgb`D0>JD5xBvhE literal 0 HcmV?d00001 diff --git a/praktikum/__pycache__/database.cpython-313.pyc b/praktikum/__pycache__/database.cpython-313.pyc new file mode 100644 index 0000000000000000000000000000000000000000..ba805a5bb2f723e496d0a7ea0f5d2631a8a627db GIT binary patch literal 2447 zcmb_dU1%It6uvV%J9B6AM@?gXwC>mnx+K{kScD>?X;RH5EH$$$lq_5(yVGRMW_G~xw%*edL$L0T^jilKLVs|fxj+t9eGaw_q$6FJMJa&_8t27XF%_YafRD^dsVI#G zyfiDP6sn{!#Tptyr;#2VLArca48f?nq$KbG7ZL)K<}AmBJu>Opu*GKVC2AHd({>Xg z8<H>}E zB9-*W2%=G4%E>t;3LdYCJjxFwiUXb6(8+-=rYp2-2`5yBrwrFvG#v8)!yzvXB<6qR z|K#7QT&-Ny;OGDB->H1!Z~Nc*w}Aga^MCQTwaOKcZu{F{xf6KZ2KjsN`cVrU@A`Ku zpI1Ij#_|F{HkSen&!he~)NTV!pe!mjZJ~7JokjCUmKHP5(|BwxaMWPwf-W|1z*wE83!G9~2<3b@k(A%QXXem&*!F3@%w?(8TNN za>>;k!^@lNF{k8FEl*8jb-gRUY!$80z~sE=7EPDof@PP$mof#YV=S2og-HPAurBLp z5;;sZ%4O3oP(BbTe^XTEgukSlFvA?Pc-i3(NMq;DpUY((lR7zrdIclr*v5*Po3hHq z(rRNPujb0sSaGctZ#7qF%#&n!jbTD8+j0qECz@j>*;Mray3-HexPtcMFJ6&ihO=$Wil+&PJ`V>Wq-S7zBBX)VmmlrX z7urPjh+P9Hrx!>N;9g(6(iit}JP0u%hRO5Kr4!w}5p84Dghq@JvRW#5kUe4DgnZ~3 z#fGJikjoZz+@fWhb_vW1A%#+&5IP7blvh1`F1#|~o6&${1B5uP;aYiOxGuF8J=b&y zxru^6f|^r4!UMwpW1S?B&(NQu7{w28Z!{|$YC?2)Q`*$OQU5}~9?C*g+jO^5m6u00#)B4#7)OauT;r!@h>cwYhdRt^lm p5!UTZIt~Wjht6|A4kLmf+(U!+(W!fA^f#1!C=UzT0|X>E{10A&{SW{E literal 0 HcmV?d00001 diff --git a/praktikum/__pycache__/ingredient.cpython-313.pyc b/praktikum/__pycache__/ingredient.cpython-313.pyc new file mode 100644 index 0000000000000000000000000000000000000000..95ca7c0179ca9a31377e2656ba223ea65a0c9047 GIT binary patch literal 1402 zcmbtUKTH!*7=PEhYp+Ke(1M|LxdyMqcaZ1HW?>3h)Ht9lxE;03M1uMUM_Y)SI2vxQvhR|ScFL?VKGx;vKeEF zDO;+kz^7XAdvPDA_~0K)N}^hRUgYI?^9k%FHRbP9^vu zL8$syP&$9hkN6(n=Lg}I!CU;0e+*Xv?t#6}55sldoYwiypZu%=Tr+$n?tJD=ynMi0 zM!3e0!k6J1+yfO$s9(tX(M>9`;2@;kMVC7$Ie{E6wNS zNI}mG$Hfw$@=B!mjY*sJXmS3@#9P6Q5^Qz0eh2F7h z%w7uIrTVfpsA0O+h%`#wiW^WGWd?(rJ|i^<;$H!^N{&Y-wvxxA$7P* zk0&9s;mn(r%@wX?*)R~7Nk+8yD!M+GiiSbOpwGAHj2wZ}ok1dv%?8)TFP#Yf_Ss(( z5%2C|B1mTN3jQI8;CXrQRv3Dl&!=$MSO*MSr*yems(WasD1A}4z243+rHd}}1JA8E zl`3o}C@ocsV90*JB6ZQL+5yA%GZFC`S~%)xj8Z?a1Gh-+AYksZdf@n!?soGLQ)T!r zgu?dvk3emZpD{(%+9_GR{$6eqSekdm(yE^zYU`4k+nG5bu(acnn%$W|JKM8Q(H0d8 qzw_{-M3700gwLV-oV2-ax%b&k;6!xXY|D})oyIa!_7?#qF8l@(gMk77 literal 0 HcmV?d00001 diff --git a/praktikum/__pycache__/ingredient_types.cpython-313.pyc b/praktikum/__pycache__/ingredient_types.cpython-313.pyc new file mode 100644 index 0000000000000000000000000000000000000000..1effe3ee8cb3cdb39a53b867982f13f682e8af97 GIT binary patch literal 383 zcmey&%ge<81W%TWW*!03k3k$5V1zP03ji5Y8G;##7=js#7=xLLn9`XwSxSq5!dw^T zU)XxF0f^f#?6}x?VfTft7xrG*abc^%#YTmTO+eQE3mY!%xv)b4$lQBjGf3%$Eg%IK z_5!8%UD%|@73>)5?5gmnVKPw9g?$&BFE(kjMbnc!*kYbYZC8l?IlUP)mk{DB*mzbRz0v7i&9fEQ}asVODYRe zi}eaBZ*kb<=BJeAq}mmMLKGBs#S%c`12ZEd<3}bIMwSm;43Y{L7({PKDPLp|FX8}7 F0su}|k$(UH literal 0 HcmV?d00001 diff --git a/bun.py b/praktikum/bun.py similarity index 100% rename from bun.py rename to praktikum/bun.py diff --git a/burger.py b/praktikum/burger.py similarity index 100% rename from burger.py rename to praktikum/burger.py diff --git a/database.py b/praktikum/database.py similarity index 100% rename from database.py rename to praktikum/database.py diff --git a/ingredient.py b/praktikum/ingredient.py similarity index 100% rename from ingredient.py rename to praktikum/ingredient.py diff --git a/ingredient_types.py b/praktikum/ingredient_types.py similarity index 100% rename from ingredient_types.py rename to praktikum/ingredient_types.py diff --git a/praktikum.py b/praktikum/praktikum.py similarity index 100% rename from praktikum.py rename to praktikum/praktikum.py diff --git a/tests/__init__.py b/tests/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/tests/__pycache__/__init__.cpython-313.pyc b/tests/__pycache__/__init__.cpython-313.pyc new file mode 100644 index 0000000000000000000000000000000000000000..f4ac749db9b489befe13a5a7cad25a86b1a9a747 GIT binary patch literal 151 zcmey&%ge<81f1p~nIQTxh=2h`DC08=kTI1Zok5e)ZzV$!6Oi{ABz4Qo*(xTqIJKxa zCb6hAB{8NrFEKkc#wD{LCqEZT#T&+yq!yPH$Hd2H=4F<|$LkeT-r}&y%}*)KNwq6t V1sVggrWnNd$jr#dSi}ru0RU1KBZvS1 literal 0 HcmV?d00001 diff --git a/tests/__pycache__/test_bun.cpython-313-pytest-8.4.1.pyc b/tests/__pycache__/test_bun.cpython-313-pytest-8.4.1.pyc new file mode 100644 index 0000000000000000000000000000000000000000..b43cdf22bb14d076d47515e54bb1699491d3fffe GIT binary patch literal 3165 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 L^2D|TgiZ9njO-ks literal 0 HcmV?d00001 diff --git a/tests/__pycache__/test_database.cpython-313-pytest-8.4.1.pyc b/tests/__pycache__/test_database.cpython-313-pytest-8.4.1.pyc new file mode 100644 index 0000000000000000000000000000000000000000..453b3f319ac711e95878e108776055a9a338ebf1 GIT binary patch literal 5110 zcmdrQU2oh(aP9Nm`96}P4QWd$)P#@Vl7<#86a|8)Xn>{=Xxu}A9Bh4ePHM%uJJvqh zrXcYpNX2s{c&)&fa}Wk4by%TB3G z6};53+*F?BrwXi~kzHhn7|L#9q{rmIE-NlbdS)GoXB{{fr|~%-i*+|n!%|&~R%=C9Xh_In}r#uF%#_c_x z-x6tc@8E?T*^bt0?&uk{x$p6)7bGKPD5I(U5TRzyfIU~q9gqZ-pBso~O04kKh}XS? z7x~DG=x)KYJ|Q(T6H@z3oG(AD-}iPhVm7Uf_$l4ov#I= z=cX@7zsnO7m!##9d+%?beOPz%=R!oA&=5PQyQ6qgc?54{zP79P>zKgo2S{{dle9vOtV(qJby5`v-ux$*(Hu-|^8aEAo}nwzZPB zTGB3;hE|j-rGfM3e06xOw0E_%7m$nP)za`9FvRamb=WWM4e+nj;gxd%dO5+4bwPfy z98;@3RLuTb9S*0|12244zc{&C(%01R_ob@)B`qMoQgxgXC6^QISQq3MCu3^0hl<(X zsQLzkfGqC{i>G(C4aGB2Lc{WTs^w{}*{NGB2kH!u<>kEF>M*U&EOUurLKzl^}DcH5aTp*2L(Fi|n(&-L%N2E9vVN+03jd zitOo4i|k5RWIu@xvA`GWN8Xah-DKAp`ddGCI zjRC4e>AcNc&#{|UvjvPy&}4IE$z+SX0IfABddyz7*eFRkAASCdH~m*n`eieu-tu2F z{c=rEFZjn__UB&n%QPbA61pIL#jjGoeDZ1`JzfJR(wVySb1@B0^dY)|Xs{k!ir&Z7 z==DN8r~Eb=gl#lv{Na6Jlv$@4d_Up2n8o4KiQol~`eCrdx3`(O=-G>%rO~Ku4^E2R zT|xK3qIlf6LHI=?L|}V^&DF<)TtC5%!3_8z!^NZJYD$u%?+JOIyw~>~8UCJBep7Zx L`+gw+#M1r*R>`;4 literal 0 HcmV?d00001 diff --git a/tests/__pycache__/test_ingredient.cpython-313-pytest-8.4.1.pyc b/tests/__pycache__/test_ingredient.cpython-313-pytest-8.4.1.pyc new file mode 100644 index 0000000000000000000000000000000000000000..472f2ab42ce54139c7442fb970c7a6a693c24267 GIT binary patch literal 6186 zcmeHL&u<%59-pzt_Vd`zw54I&Y+y~&qL?P`50U}}H>pY}O{?xIQzK9!jV8_{ovmY= zH)BD9kXV6$!(O@fu!m$%9QKdcqphS(^&|umr{2PPC8VDCe&3s4PFz9=aO#u&o$v4W zW8Qq``=*o2sS>n*O&zyCA(Hex3Vu^Ort=OkUrD;8%Qqw{^J{D-?!@xaj6xH6VM)>? zSYkI)GlUYZi{D7ks8pTF&`e%BAq`2oG9c-RiI~Vsv&(WJ`3Qwu1(~VWYs=JHw5^(3 zh_U|bx30}jzj=N7)@|eVdw-iY^eg47)9g>X)o)$Dc@uOMky6l%Cv^TJ(w0Dn{;}d9sR#VE# zthcp=M$meBDA)}3rE>hz+mclOV8nB|CNIL&=7Qd4lhIhg!-FH2Q{%E8d$%$koA7+~ z_#8a2qV?L;VXv=))8e1Iddd2c;dMwVD-+(#{Lz`uc187t-aZ$N`l}3M3TM*WJK^`S zuV>Qt)Ro_Srg1;_fo2*+!AuoBQB=m@Zqukr^KIb>S6OF71!P zzghsUCApR;$&KY@sX%UR$d6;CQbAsO`}9a-b-duHA9+l^lt=D@uL8zXohvxSWy>}A zQ=xu5Qcyq6mxTAlg2UACAz}&cv!SqfWAz-83rNP1TzuSTI*vtM-p#8J2+CIrD#h*2 z(m@AIoocMQ@UR*tWk-#QS*;q>YS2o3rD0NwWx;GLH0`Qu*Bq9t*1^MJCk*FKy;)u4 zuEuh;zF<}je3)6K=as1Nnu9Nne9&0w>ZSQIERZvA(&nN$ z@6^n@*8H1xqgr3_F2?zJTp5Q`BOG9{vC8`SnuR)XyV$92+$;9q=W2-6q~|6D_9#Q84$( zD1wIP(hemsKptV~8M%e|qmdCQP(L7}^el|IH)#Cx(3M#DcHRRX@Y0wI>af|M%7Vdo zEv(4O$uLhBj0DGw*`BqQe>@0kv#`J3byM%!EQ~thk1f3Jsm=ED4r{ahJfpSQL1qej zJ+iVrXBw{O9yl53itah%z0p4lEBL(f78w;jD7)G(s?|8GZ2T%%!NbfGUx-LR4o zMHq(_45i%Z(AYAt8(T0plw)wBVugJWCUiv*bG`-w#YuKgaUyW0N&{R2{ZiZQDb<^E+BaY$t57H-!L2(4!sq_bise23HQP<=o{#Og%U3i zDV_~E!930ae9-w^`Yw_AV(uSv+lj$;VzBkx>kre<-^;A)Epi^Y?0Tg|E+Cg%cUt7- zhw1#ik@c%Bat^tdJZt`5fk*7*QkRn*DV0q9a3q#E+c^r{4stuHl+6B4-HZj@j2*O_ zos@#(RbaeoEOEJ$0d5DmUWtBo;$4ZRWS*%5nWkj1{J8Rk93%e64qoE3lZLq8Me*qY zK3Np0^#QeA_|y@oSjy8;{)AqEOze1rns;6MZgZs=ekBku{()X5{?)?~r2yYUd?$48 x4o2P(+w?NM0e!&ckQ_XN?8IeR{#KGclRnLTBc1tHD*vdA$y4(GB&2+LKLM)+ZE64j literal 0 HcmV?d00001 diff --git a/tests/test_bun.py b/tests/test_bun.py new file mode 100644 index 000000000..a11e57b90 --- /dev/null +++ b/tests/test_bun.py @@ -0,0 +1,17 @@ +import pytest +from praktikum.bun import Bun + +class TestBun: + @pytest.mark.parametrize ("name, price",[("black bun", 100), + ("white bun", 200), + ("red bun", 300) ]) + def test_bun_name (self, name, price): + bun = Bun(name, price) + assert bun.get_name() == name + + @pytest.mark.parametrize ("name, price",[("black bun", 100), + ("white bun", 200), + ("red bun", 300) ]) + def test_bun_price (self, name, price): + bun = Bun(name, price) + assert bun.get_price() == price \ No newline at end of file diff --git a/tests/test_burger.py b/tests/test_burger.py new file mode 100644 index 000000000..6d12e69a3 --- /dev/null +++ b/tests/test_burger.py @@ -0,0 +1,50 @@ +from praktikum.burger import Burger + +class TestBurger: + def test_create_burger(self): + burger = Burger() + assert burger.bun is None and burger.ingredients == [] + + def test_set_bun(self, mock_bun): + burger = Burger() + burger.set_buns(mock_bun) + assert burger.bun == mock_bun + + def test_add_ingredient(self, mock_filling): + burger = Burger() + burger.add_ingredient(mock_filling) + assert mock_filling in burger.ingredients + + def test_remove_ingredient(self, mock_sauce): + burger = Burger() + burger.add_ingredient(mock_sauce) + burger.remove_ingredient(0) + assert mock_sauce not in burger.ingredients + + def test_move_ingredients(self, mock_filling, mock_sauce): + burger = Burger() + burger.add_ingredient(mock_filling) + burger.add_ingredient(mock_sauce) + burger.move_ingredient(1, 0) + assert burger.ingredients[0] == mock_sauce and burger.ingredients[1] == mock_filling + + def test_get_price(self, mock_bun, mock_filling, mock_sauce): + burger = Burger() + burger.set_buns(mock_bun) + burger.add_ingredient(mock_filling) + burger.add_ingredient(mock_sauce) + expected_price = (mock_bun.get_price() * 2 + mock_filling.get_price() + mock_sauce.get_price()) + assert burger.get_price() == expected_price + + def test_get_receipt(self, mock_bun, mock_sauce, mock_filling): + burger = Burger() + burger.set_buns(mock_bun) + burger.add_ingredient(mock_sauce) + burger.add_ingredient(mock_filling) + expected_receipt = ("(==== yellow bun ====)\n" + "= sauce teriyaki =\n" + "= filling snake =\n" + "(==== yellow bun ====)\n" + "\n" + f"Price: {burger.get_price()}") + assert burger.get_receipt() == expected_receipt \ No newline at end of file diff --git a/tests/test_database.py b/tests/test_database.py new file mode 100644 index 000000000..292957de9 --- /dev/null +++ b/tests/test_database.py @@ -0,0 +1,30 @@ +import pytest +from praktikum.database import Database +from praktikum.ingredient_types import INGREDIENT_TYPE_FILLING, INGREDIENT_TYPE_SAUCE + +class TestDatabase: + + @pytest.mark.parametrize("index, name, price", [(0, "black bun", 100), + (1, "white bun", 200), + (2, "red bun", 300)]) + def test_available_buns(self, index, name, price): + db = Database() + buns = db.available_buns() + assert buns[index].name == name + assert buns[index].price == price + + @pytest.mark.parametrize("index, type, name, price", [(0, INGREDIENT_TYPE_SAUCE, "hot sauce", 100), + (1, INGREDIENT_TYPE_SAUCE, "sour cream", 200), + (2, INGREDIENT_TYPE_SAUCE, "chili sauce", 300), + (3, INGREDIENT_TYPE_FILLING, "cutlet", 100), + (4, INGREDIENT_TYPE_FILLING, "dinosaur", 200), + (5, INGREDIENT_TYPE_FILLING, "sausage", 300)]) + def test_available_ingredients(self, index, type, name, price): + db = Database() + ingr = db.available_ingredients() + assert ingr[index].type == type + assert ingr[index].name == name + assert ingr[index].price == price + + + \ No newline at end of file diff --git a/tests/test_ingredient.py b/tests/test_ingredient.py new file mode 100644 index 000000000..5f5084fe8 --- /dev/null +++ b/tests/test_ingredient.py @@ -0,0 +1,50 @@ +import pytest +from praktikum.ingredient import Ingredient +from praktikum.ingredient_types import INGREDIENT_TYPE_SAUCE, INGREDIENT_TYPE_FILLING + +class TestIngredient: + + @pytest.mark.parametrize("ingredient_type, name, price", [ + (INGREDIENT_TYPE_SAUCE, "hot sauce", 100), + (INGREDIENT_TYPE_SAUCE, "sour cream", 200), + (INGREDIENT_TYPE_SAUCE, "chili sauce", 300), + (INGREDIENT_TYPE_FILLING, "cutlet", 100), + (INGREDIENT_TYPE_FILLING, "dinosaur", 200), + (INGREDIENT_TYPE_FILLING, "sausage", 300) + ]) + def test_ingredient_price(self, ingredient_type, name, price): + ingredient = Ingredient(ingredient_type, name, price) + assert ingredient.get_price() == price + + + @pytest.mark.parametrize("ingredient_type, name, price", [ + (INGREDIENT_TYPE_SAUCE, "hot sauce", 100), + (INGREDIENT_TYPE_SAUCE, "sour cream", 200), + (INGREDIENT_TYPE_SAUCE, "chili sauce", 300), + (INGREDIENT_TYPE_FILLING, "cutlet", 100), + (INGREDIENT_TYPE_FILLING, "dinosaur", 200), + (INGREDIENT_TYPE_FILLING, "sausage", 300) + ]) + def test_ingredient_name(self, ingredient_type, name, price): + ingredient = Ingredient(ingredient_type, name, price) + assert ingredient.get_name() == name + + + @pytest.mark.parametrize("ingredient_type, name, price", [ + (INGREDIENT_TYPE_SAUCE, "hot sauce", 100), + (INGREDIENT_TYPE_SAUCE, "sour cream", 200), + (INGREDIENT_TYPE_SAUCE, "chili sauce", 300) + ]) + def test_ingredient_type_sauce(self, ingredient_type, name, price): + ingredient = Ingredient(ingredient_type, name, price) + assert ingredient.get_type() == INGREDIENT_TYPE_SAUCE + + + @pytest.mark.parametrize("ingredient_type, name, price", [ + (INGREDIENT_TYPE_FILLING, "cutlet", 100), + (INGREDIENT_TYPE_FILLING, "dinosaur", 200), + (INGREDIENT_TYPE_FILLING, "sausage", 300) + ]) + def test_ingredient_type_filling(self, ingredient_type, name, price): + ingredient = Ingredient(ingredient_type, name, price) + assert ingredient.get_type() == INGREDIENT_TYPE_FILLING \ No newline at end of file From 28353de37b81dea6f89611e7af3389b3cc4a8459 Mon Sep 17 00:00:00 2001 From: Kyber-strannik Date: Mon, 17 Nov 2025 13:54:38 +0300 Subject: [PATCH 2/2] Diplom_unit_for_burgers_and_htmlcov --- htmlcov/class_index.html | 195 +++++ htmlcov/coverage_html_cb_497bf287.js | 733 ++++++++++++++++++ htmlcov/favicon_32_cb_58284776.png | Bin 0 -> 1732 bytes htmlcov/function_index.html | 307 ++++++++ htmlcov/index.html | 153 ++++ htmlcov/keybd_closed_cb_ce680311.png | Bin 0 -> 9004 bytes htmlcov/status.json | 1 + htmlcov/style_cb_db813965.css | 337 ++++++++ htmlcov/z_c68eb0c7512457e4___init___py.html | 97 +++ htmlcov/z_c68eb0c7512457e4_bun_py.html | 112 +++ htmlcov/z_c68eb0c7512457e4_burger_py.html | 145 ++++ htmlcov/z_c68eb0c7512457e4_database_py.html | 130 ++++ htmlcov/z_c68eb0c7512457e4_ingredient_py.html | 117 +++ ..._c68eb0c7512457e4_ingredient_types_py.html | 104 +++ htmlcov/z_c68eb0c7512457e4_praktikum_py.html | 138 ++++ 15 files changed, 2569 insertions(+) create mode 100644 htmlcov/class_index.html create mode 100644 htmlcov/coverage_html_cb_497bf287.js create mode 100644 htmlcov/favicon_32_cb_58284776.png create mode 100644 htmlcov/function_index.html create mode 100644 htmlcov/index.html create mode 100644 htmlcov/keybd_closed_cb_ce680311.png create mode 100644 htmlcov/status.json create mode 100644 htmlcov/style_cb_db813965.css create mode 100644 htmlcov/z_c68eb0c7512457e4___init___py.html create mode 100644 htmlcov/z_c68eb0c7512457e4_bun_py.html create mode 100644 htmlcov/z_c68eb0c7512457e4_burger_py.html create mode 100644 htmlcov/z_c68eb0c7512457e4_database_py.html create mode 100644 htmlcov/z_c68eb0c7512457e4_ingredient_py.html create mode 100644 htmlcov/z_c68eb0c7512457e4_ingredient_types_py.html create mode 100644 htmlcov/z_c68eb0c7512457e4_praktikum_py.html diff --git a/htmlcov/class_index.html b/htmlcov/class_index.html new file mode 100644 index 000000000..71a9e18d0 --- /dev/null +++ b/htmlcov/class_index.html @@ -0,0 +1,195 @@ + + + + + Coverage report + + + + + +
+
+

Coverage report: + 78% +

+ +
+ +
+ + +
+
+

+ Files + Functions + Classes +

+

+ coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+
+
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Fileclassstatementsmissingexcludedcoverage
praktikum\__init__.py(no class)000100%
praktikum\bun.pyBun400100%
praktikum\bun.py(no class)400100%
praktikum\burger.pyBurger1600100%
praktikum\burger.py(no class)1100100%
praktikum\database.pyDatabase1300100%
praktikum\database.py(no class)800100%
praktikum\ingredient.pyIngredient600100%
praktikum\ingredient.py(no class)500100%
praktikum\ingredient_types.py(no class)200100%
praktikum\praktikum.py(no class)202000%
Total 8920078%
+

+ No items found using the specified filter. +

+
+ + + diff --git a/htmlcov/coverage_html_cb_497bf287.js b/htmlcov/coverage_html_cb_497bf287.js new file mode 100644 index 000000000..1face13de --- /dev/null +++ b/htmlcov/coverage_html_cb_497bf287.js @@ -0,0 +1,733 @@ +// Licensed under the Apache License: http://www.apache.org/licenses/LICENSE-2.0 +// For details: https://github.com/nedbat/coveragepy/blob/master/NOTICE.txt + +// Coverage.py HTML report browser code. +/*jslint browser: true, sloppy: true, vars: true, plusplus: true, maxerr: 50, indent: 4 */ +/*global coverage: true, document, window, $ */ + +coverage = {}; + +// General helpers +function debounce(callback, wait) { + let timeoutId = null; + return function(...args) { + clearTimeout(timeoutId); + timeoutId = setTimeout(() => { + callback.apply(this, args); + }, wait); + }; +}; + +function checkVisible(element) { + const rect = element.getBoundingClientRect(); + const viewBottom = Math.max(document.documentElement.clientHeight, window.innerHeight); + const viewTop = 30; + return !(rect.bottom < viewTop || rect.top >= viewBottom); +} + +function on_click(sel, fn) { + const elt = document.querySelector(sel); + if (elt) { + elt.addEventListener("click", fn); + } +} + +// Helpers for table sorting +function getCellValue(row, column = 0) { + const cell = row.cells[column] // nosemgrep: eslint.detect-object-injection + if (cell.childElementCount == 1) { + var child = cell.firstElementChild; + if (child.tagName === "A") { + child = child.firstElementChild; + } + if (child instanceof HTMLDataElement && child.value) { + return child.value; + } + } + return cell.innerText || cell.textContent; +} + +function rowComparator(rowA, rowB, column = 0) { + let valueA = getCellValue(rowA, column); + let valueB = getCellValue(rowB, column); + if (!isNaN(valueA) && !isNaN(valueB)) { + return valueA - valueB; + } + return valueA.localeCompare(valueB, undefined, {numeric: true}); +} + +function sortColumn(th) { + // Get the current sorting direction of the selected header, + // clear state on other headers and then set the new sorting direction. + const currentSortOrder = th.getAttribute("aria-sort"); + [...th.parentElement.cells].forEach(header => header.setAttribute("aria-sort", "none")); + var direction; + if (currentSortOrder === "none") { + direction = th.dataset.defaultSortOrder || "ascending"; + } + else if (currentSortOrder === "ascending") { + direction = "descending"; + } + else { + direction = "ascending"; + } + th.setAttribute("aria-sort", direction); + + const column = [...th.parentElement.cells].indexOf(th) + + // Sort all rows and afterwards append them in order to move them in the DOM. + Array.from(th.closest("table").querySelectorAll("tbody tr")) + .sort((rowA, rowB) => rowComparator(rowA, rowB, column) * (direction === "ascending" ? 1 : -1)) + .forEach(tr => tr.parentElement.appendChild(tr)); + + // Save the sort order for next time. + if (th.id !== "region") { + let th_id = "file"; // Sort by file if we don't have a column id + let current_direction = direction; + const stored_list = localStorage.getItem(coverage.INDEX_SORT_STORAGE); + if (stored_list) { + ({th_id, direction} = JSON.parse(stored_list)) + } + localStorage.setItem(coverage.INDEX_SORT_STORAGE, JSON.stringify({ + "th_id": th.id, + "direction": current_direction + })); + if (th.id !== th_id || document.getElementById("region")) { + // Sort column has changed, unset sorting by function or class. + localStorage.setItem(coverage.SORTED_BY_REGION, JSON.stringify({ + "by_region": false, + "region_direction": current_direction + })); + } + } + else { + // Sort column has changed to by function or class, remember that. + localStorage.setItem(coverage.SORTED_BY_REGION, JSON.stringify({ + "by_region": true, + "region_direction": direction + })); + } +} + +// Find all the elements with data-shortcut attribute, and use them to assign a shortcut key. +coverage.assign_shortkeys = function () { + document.querySelectorAll("[data-shortcut]").forEach(element => { + document.addEventListener("keypress", event => { + if (event.target.tagName.toLowerCase() === "input") { + return; // ignore keypress from search filter + } + if (event.key === element.dataset.shortcut) { + element.click(); + } + }); + }); +}; + +// Create the events for the filter box. +coverage.wire_up_filter = function () { + // Populate the filter and hide100 inputs if there are saved values for them. + const saved_filter_value = localStorage.getItem(coverage.FILTER_STORAGE); + if (saved_filter_value) { + document.getElementById("filter").value = saved_filter_value; + } + const saved_hide100_value = localStorage.getItem(coverage.HIDE100_STORAGE); + if (saved_hide100_value) { + document.getElementById("hide100").checked = JSON.parse(saved_hide100_value); + } + + // Cache elements. + const table = document.querySelector("table.index"); + const table_body_rows = table.querySelectorAll("tbody tr"); + const no_rows = document.getElementById("no_rows"); + + // Observe filter keyevents. + const filter_handler = (event => { + // Keep running total of each metric, first index contains number of shown rows + const totals = new Array(table.rows[0].cells.length).fill(0); + // Accumulate the percentage as fraction + totals[totals.length - 1] = { "numer": 0, "denom": 0 }; // nosemgrep: eslint.detect-object-injection + + var text = document.getElementById("filter").value; + // Store filter value + localStorage.setItem(coverage.FILTER_STORAGE, text); + const casefold = (text === text.toLowerCase()); + const hide100 = document.getElementById("hide100").checked; + // Store hide value. + localStorage.setItem(coverage.HIDE100_STORAGE, JSON.stringify(hide100)); + + // Hide / show elements. + table_body_rows.forEach(row => { + var show = false; + // Check the text filter. + for (let column = 0; column < totals.length; column++) { + cell = row.cells[column]; + if (cell.classList.contains("name")) { + var celltext = cell.textContent; + if (casefold) { + celltext = celltext.toLowerCase(); + } + if (celltext.includes(text)) { + show = true; + } + } + } + + // Check the "hide covered" filter. + if (show && hide100) { + const [numer, denom] = row.cells[row.cells.length - 1].dataset.ratio.split(" "); + show = (numer !== denom); + } + + if (!show) { + // hide + row.classList.add("hidden"); + return; + } + + // show + row.classList.remove("hidden"); + totals[0]++; + + for (let column = 0; column < totals.length; column++) { + // Accumulate dynamic totals + cell = row.cells[column] // nosemgrep: eslint.detect-object-injection + if (cell.classList.contains("name")) { + continue; + } + if (column === totals.length - 1) { + // Last column contains percentage + const [numer, denom] = cell.dataset.ratio.split(" "); + totals[column]["numer"] += parseInt(numer, 10); // nosemgrep: eslint.detect-object-injection + totals[column]["denom"] += parseInt(denom, 10); // nosemgrep: eslint.detect-object-injection + } + else { + totals[column] += parseInt(cell.textContent, 10); // nosemgrep: eslint.detect-object-injection + } + } + }); + + // Show placeholder if no rows will be displayed. + if (!totals[0]) { + // Show placeholder, hide table. + no_rows.style.display = "block"; + table.style.display = "none"; + return; + } + + // Hide placeholder, show table. + no_rows.style.display = null; + table.style.display = null; + + const footer = table.tFoot.rows[0]; + // Calculate new dynamic sum values based on visible rows. + for (let column = 0; column < totals.length; column++) { + // Get footer cell element. + const cell = footer.cells[column]; // nosemgrep: eslint.detect-object-injection + if (cell.classList.contains("name")) { + continue; + } + + // Set value into dynamic footer cell element. + if (column === totals.length - 1) { + // Percentage column uses the numerator and denominator, + // and adapts to the number of decimal places. + const match = /\.([0-9]+)/.exec(cell.textContent); + const places = match ? match[1].length : 0; + const { numer, denom } = totals[column]; // nosemgrep: eslint.detect-object-injection + cell.dataset.ratio = `${numer} ${denom}`; + // Check denom to prevent NaN if filtered files contain no statements + cell.textContent = denom + ? `${(numer * 100 / denom).toFixed(places)}%` + : `${(100).toFixed(places)}%`; + } + else { + cell.textContent = totals[column]; // nosemgrep: eslint.detect-object-injection + } + } + }); + + document.getElementById("filter").addEventListener("input", debounce(filter_handler)); + document.getElementById("hide100").addEventListener("input", debounce(filter_handler)); + + // Trigger change event on setup, to force filter on page refresh + // (filter value may still be present). + document.getElementById("filter").dispatchEvent(new Event("input")); + document.getElementById("hide100").dispatchEvent(new Event("input")); +}; +coverage.FILTER_STORAGE = "COVERAGE_FILTER_VALUE"; +coverage.HIDE100_STORAGE = "COVERAGE_HIDE100_VALUE"; + +// Set up the click-to-sort columns. +coverage.wire_up_sorting = function () { + document.querySelectorAll("[data-sortable] th[aria-sort]").forEach( + th => th.addEventListener("click", e => sortColumn(e.target)) + ); + + // Look for a localStorage item containing previous sort settings: + let th_id = "file", direction = "ascending"; + const stored_list = localStorage.getItem(coverage.INDEX_SORT_STORAGE); + if (stored_list) { + ({th_id, direction} = JSON.parse(stored_list)); + } + let by_region = false, region_direction = "ascending"; + const sorted_by_region = localStorage.getItem(coverage.SORTED_BY_REGION); + if (sorted_by_region) { + ({ + by_region, + region_direction + } = JSON.parse(sorted_by_region)); + } + + const region_id = "region"; + if (by_region && document.getElementById(region_id)) { + direction = region_direction; + } + // If we are in a page that has a column with id of "region", sort on + // it if the last sort was by function or class. + let th; + if (document.getElementById(region_id)) { + th = document.getElementById(by_region ? region_id : th_id); + } + else { + th = document.getElementById(th_id); + } + th.setAttribute("aria-sort", direction === "ascending" ? "descending" : "ascending"); + th.click() +}; + +coverage.INDEX_SORT_STORAGE = "COVERAGE_INDEX_SORT_2"; +coverage.SORTED_BY_REGION = "COVERAGE_SORT_REGION"; + +// Loaded on index.html +coverage.index_ready = function () { + coverage.assign_shortkeys(); + coverage.wire_up_filter(); + coverage.wire_up_sorting(); + + on_click(".button_prev_file", coverage.to_prev_file); + on_click(".button_next_file", coverage.to_next_file); + + on_click(".button_show_hide_help", coverage.show_hide_help); +}; + +// -- pyfile stuff -- + +coverage.LINE_FILTERS_STORAGE = "COVERAGE_LINE_FILTERS"; + +coverage.pyfile_ready = function () { + // If we're directed to a particular line number, highlight the line. + var frag = location.hash; + if (frag.length > 2 && frag[1] === "t") { + document.querySelector(frag).closest(".n").classList.add("highlight"); + coverage.set_sel(parseInt(frag.substr(2), 10)); + } + else { + coverage.set_sel(0); + } + + on_click(".button_toggle_run", coverage.toggle_lines); + on_click(".button_toggle_mis", coverage.toggle_lines); + on_click(".button_toggle_exc", coverage.toggle_lines); + on_click(".button_toggle_par", coverage.toggle_lines); + + on_click(".button_next_chunk", coverage.to_next_chunk_nicely); + on_click(".button_prev_chunk", coverage.to_prev_chunk_nicely); + on_click(".button_top_of_page", coverage.to_top); + on_click(".button_first_chunk", coverage.to_first_chunk); + + on_click(".button_prev_file", coverage.to_prev_file); + on_click(".button_next_file", coverage.to_next_file); + on_click(".button_to_index", coverage.to_index); + + on_click(".button_show_hide_help", coverage.show_hide_help); + + coverage.filters = undefined; + try { + coverage.filters = localStorage.getItem(coverage.LINE_FILTERS_STORAGE); + } catch(err) {} + + if (coverage.filters) { + coverage.filters = JSON.parse(coverage.filters); + } + else { + coverage.filters = {run: false, exc: true, mis: true, par: true}; + } + + for (cls in coverage.filters) { + coverage.set_line_visibilty(cls, coverage.filters[cls]); // nosemgrep: eslint.detect-object-injection + } + + coverage.assign_shortkeys(); + coverage.init_scroll_markers(); + coverage.wire_up_sticky_header(); + + document.querySelectorAll("[id^=ctxs]").forEach( + cbox => cbox.addEventListener("click", coverage.expand_contexts) + ); + + // Rebuild scroll markers when the window height changes. + window.addEventListener("resize", coverage.build_scroll_markers); +}; + +coverage.toggle_lines = function (event) { + const btn = event.target.closest("button"); + const category = btn.value + const show = !btn.classList.contains("show_" + category); + coverage.set_line_visibilty(category, show); + coverage.build_scroll_markers(); + coverage.filters[category] = show; + try { + localStorage.setItem(coverage.LINE_FILTERS_STORAGE, JSON.stringify(coverage.filters)); + } catch(err) {} +}; + +coverage.set_line_visibilty = function (category, should_show) { + const cls = "show_" + category; + const btn = document.querySelector(".button_toggle_" + category); + if (btn) { + if (should_show) { + document.querySelectorAll("#source ." + category).forEach(e => e.classList.add(cls)); + btn.classList.add(cls); + } + else { + document.querySelectorAll("#source ." + category).forEach(e => e.classList.remove(cls)); + btn.classList.remove(cls); + } + } +}; + +// Return the nth line div. +coverage.line_elt = function (n) { + return document.getElementById("t" + n)?.closest("p"); +}; + +// Set the selection. b and e are line numbers. +coverage.set_sel = function (b, e) { + // The first line selected. + coverage.sel_begin = b; + // The next line not selected. + coverage.sel_end = (e === undefined) ? b+1 : e; +}; + +coverage.to_top = function () { + coverage.set_sel(0, 1); + coverage.scroll_window(0); +}; + +coverage.to_first_chunk = function () { + coverage.set_sel(0, 1); + coverage.to_next_chunk(); +}; + +coverage.to_prev_file = function () { + window.location = document.getElementById("prevFileLink").href; +} + +coverage.to_next_file = function () { + window.location = document.getElementById("nextFileLink").href; +} + +coverage.to_index = function () { + location.href = document.getElementById("indexLink").href; +} + +coverage.show_hide_help = function () { + const helpCheck = document.getElementById("help_panel_state") + helpCheck.checked = !helpCheck.checked; +} + +// Return a string indicating what kind of chunk this line belongs to, +// or null if not a chunk. +coverage.chunk_indicator = function (line_elt) { + const classes = line_elt?.className; + if (!classes) { + return null; + } + const match = classes.match(/\bshow_\w+\b/); + if (!match) { + return null; + } + return match[0]; +}; + +coverage.to_next_chunk = function () { + const c = coverage; + + // Find the start of the next colored chunk. + var probe = c.sel_end; + var chunk_indicator, probe_line; + while (true) { + probe_line = c.line_elt(probe); + if (!probe_line) { + return; + } + chunk_indicator = c.chunk_indicator(probe_line); + if (chunk_indicator) { + break; + } + probe++; + } + + // There's a next chunk, `probe` points to it. + var begin = probe; + + // Find the end of this chunk. + var next_indicator = chunk_indicator; + while (next_indicator === chunk_indicator) { + probe++; + probe_line = c.line_elt(probe); + next_indicator = c.chunk_indicator(probe_line); + } + c.set_sel(begin, probe); + c.show_selection(); +}; + +coverage.to_prev_chunk = function () { + const c = coverage; + + // Find the end of the prev colored chunk. + var probe = c.sel_begin-1; + var probe_line = c.line_elt(probe); + if (!probe_line) { + return; + } + var chunk_indicator = c.chunk_indicator(probe_line); + while (probe > 1 && !chunk_indicator) { + probe--; + probe_line = c.line_elt(probe); + if (!probe_line) { + return; + } + chunk_indicator = c.chunk_indicator(probe_line); + } + + // There's a prev chunk, `probe` points to its last line. + var end = probe+1; + + // Find the beginning of this chunk. + var prev_indicator = chunk_indicator; + while (prev_indicator === chunk_indicator) { + probe--; + if (probe <= 0) { + return; + } + probe_line = c.line_elt(probe); + prev_indicator = c.chunk_indicator(probe_line); + } + c.set_sel(probe+1, end); + c.show_selection(); +}; + +// Returns 0, 1, or 2: how many of the two ends of the selection are on +// the screen right now? +coverage.selection_ends_on_screen = function () { + if (coverage.sel_begin === 0) { + return 0; + } + + const begin = coverage.line_elt(coverage.sel_begin); + const end = coverage.line_elt(coverage.sel_end-1); + + return ( + (checkVisible(begin) ? 1 : 0) + + (checkVisible(end) ? 1 : 0) + ); +}; + +coverage.to_next_chunk_nicely = function () { + if (coverage.selection_ends_on_screen() === 0) { + // The selection is entirely off the screen: + // Set the top line on the screen as selection. + + // This will select the top-left of the viewport + // As this is most likely the span with the line number we take the parent + const line = document.elementFromPoint(0, 0).parentElement; + if (line.parentElement !== document.getElementById("source")) { + // The element is not a source line but the header or similar + coverage.select_line_or_chunk(1); + } + else { + // We extract the line number from the id + coverage.select_line_or_chunk(parseInt(line.id.substring(1), 10)); + } + } + coverage.to_next_chunk(); +}; + +coverage.to_prev_chunk_nicely = function () { + if (coverage.selection_ends_on_screen() === 0) { + // The selection is entirely off the screen: + // Set the lowest line on the screen as selection. + + // This will select the bottom-left of the viewport + // As this is most likely the span with the line number we take the parent + const line = document.elementFromPoint(document.documentElement.clientHeight-1, 0).parentElement; + if (line.parentElement !== document.getElementById("source")) { + // The element is not a source line but the header or similar + coverage.select_line_or_chunk(coverage.lines_len); + } + else { + // We extract the line number from the id + coverage.select_line_or_chunk(parseInt(line.id.substring(1), 10)); + } + } + coverage.to_prev_chunk(); +}; + +// Select line number lineno, or if it is in a colored chunk, select the +// entire chunk +coverage.select_line_or_chunk = function (lineno) { + var c = coverage; + var probe_line = c.line_elt(lineno); + if (!probe_line) { + return; + } + var the_indicator = c.chunk_indicator(probe_line); + if (the_indicator) { + // The line is in a highlighted chunk. + // Search backward for the first line. + var probe = lineno; + var indicator = the_indicator; + while (probe > 0 && indicator === the_indicator) { + probe--; + probe_line = c.line_elt(probe); + if (!probe_line) { + break; + } + indicator = c.chunk_indicator(probe_line); + } + var begin = probe + 1; + + // Search forward for the last line. + probe = lineno; + indicator = the_indicator; + while (indicator === the_indicator) { + probe++; + probe_line = c.line_elt(probe); + indicator = c.chunk_indicator(probe_line); + } + + coverage.set_sel(begin, probe); + } + else { + coverage.set_sel(lineno); + } +}; + +coverage.show_selection = function () { + // Highlight the lines in the chunk + document.querySelectorAll("#source .highlight").forEach(e => e.classList.remove("highlight")); + for (let probe = coverage.sel_begin; probe < coverage.sel_end; probe++) { + coverage.line_elt(probe).querySelector(".n").classList.add("highlight"); + } + + coverage.scroll_to_selection(); +}; + +coverage.scroll_to_selection = function () { + // Scroll the page if the chunk isn't fully visible. + if (coverage.selection_ends_on_screen() < 2) { + const element = coverage.line_elt(coverage.sel_begin); + coverage.scroll_window(element.offsetTop - 60); + } +}; + +coverage.scroll_window = function (to_pos) { + window.scroll({top: to_pos, behavior: "smooth"}); +}; + +coverage.init_scroll_markers = function () { + // Init some variables + coverage.lines_len = document.querySelectorAll("#source > p").length; + + // Build html + coverage.build_scroll_markers(); +}; + +coverage.build_scroll_markers = function () { + const temp_scroll_marker = document.getElementById("scroll_marker") + if (temp_scroll_marker) temp_scroll_marker.remove(); + // Don't build markers if the window has no scroll bar. + if (document.body.scrollHeight <= window.innerHeight) { + return; + } + + const marker_scale = window.innerHeight / document.body.scrollHeight; + const line_height = Math.min(Math.max(3, window.innerHeight / coverage.lines_len), 10); + + let previous_line = -99, last_mark, last_top; + + const scroll_marker = document.createElement("div"); + scroll_marker.id = "scroll_marker"; + document.getElementById("source").querySelectorAll( + "p.show_run, p.show_mis, p.show_exc, p.show_exc, p.show_par" + ).forEach(element => { + const line_top = Math.floor(element.offsetTop * marker_scale); + const line_number = parseInt(element.querySelector(".n a").id.substr(1)); + + if (line_number === previous_line + 1) { + // If this solid missed block just make previous mark higher. + last_mark.style.height = `${line_top + line_height - last_top}px`; + } + else { + // Add colored line in scroll_marker block. + last_mark = document.createElement("div"); + last_mark.id = `m${line_number}`; + last_mark.classList.add("marker"); + last_mark.style.height = `${line_height}px`; + last_mark.style.top = `${line_top}px`; + scroll_marker.append(last_mark); + last_top = line_top; + } + + previous_line = line_number; + }); + + // Append last to prevent layout calculation + document.body.append(scroll_marker); +}; + +coverage.wire_up_sticky_header = function () { + const header = document.querySelector("header"); + const header_bottom = ( + header.querySelector(".content h2").getBoundingClientRect().top - + header.getBoundingClientRect().top + ); + + function updateHeader() { + if (window.scrollY > header_bottom) { + header.classList.add("sticky"); + } + else { + header.classList.remove("sticky"); + } + } + + window.addEventListener("scroll", updateHeader); + updateHeader(); +}; + +coverage.expand_contexts = function (e) { + var ctxs = e.target.parentNode.querySelector(".ctxs"); + + if (!ctxs.classList.contains("expanded")) { + var ctxs_text = ctxs.textContent; + var width = Number(ctxs_text[0]); + ctxs.textContent = ""; + for (var i = 1; i < ctxs_text.length; i += width) { + key = ctxs_text.substring(i, i + width).trim(); + ctxs.appendChild(document.createTextNode(contexts[key])); + ctxs.appendChild(document.createElement("br")); + } + ctxs.classList.add("expanded"); + } +}; + +document.addEventListener("DOMContentLoaded", () => { + if (document.body.classList.contains("indexfile")) { + coverage.index_ready(); + } + else { + coverage.pyfile_ready(); + } +}); diff --git a/htmlcov/favicon_32_cb_58284776.png b/htmlcov/favicon_32_cb_58284776.png new file mode 100644 index 0000000000000000000000000000000000000000..8649f0475d8d20793b2ec431fe25a186a414cf10 GIT binary patch literal 1732 zcmV;#20QtQP)K2KOkBOVxIZChq#W-v7@TU%U6P(wycKT1hUJUToW3ke1U1ONa4 z000000000000000bb)GRa9mqwR9|UWHy;^RUrt?IT__Y0JUcxmBP0(51q1>E00030 z|NrOz)aw7%8sJzM<5^g%z7^qE`}_Ot|JUUG(NUkWzR|7K?Zo%@_v-8G-1N%N=D$;; zw;keH4dGY$`1t4M=HK_s*zm^0#KgqfwWhe3qO_HtvXYvtjgX>;-~C$L`&k>^R)9)7 zdPh2TL^pCnHC#0+_4D)M`p?qp!pq{jO_{8;$fbaflbx`Tn52n|n}8VFRTA1&ugOP< zPd{uvFjz7t*Vot1&d$l-xWCk}s;sQL&#O(Bskh6gqNJv>#iB=ypG1e3K!K4yc7!~M zfj4S*g^zZ7eP$+_Sl07Z646l;%urinP#D8a6TwRtnLIRcI!r4f@bK~9-`~;E(N?Lv zSEst7s;rcxsi~}{Nsytfz@MtUoR*iFc8!#vvx}Umhm4blk(_~MdVD-@dW&>!Nn~ro z_E~-ESVQAj6Wmn;(olz(O&_{U2*pZBc1aYjMh>Dq3z|6`jW`RDHV=t3I6yRKJ~LOX zz_z!!vbVXPqob#=pj3^VMT?x6t(irRmSKsMo1~LLkB&=#j!=M%NP35mfqim$drWb9 zYIb>no_LUwc!r^NkDzs4YHu@=ZHRzrafWDZd1EhEVq=tGX?tK$pIa)DTh#bkvh!J- z?^%@YS!U*0E8$q$_*aOTQ&)Ra64g>ep;BdcQgvlg8qQHrP*E$;P{-m=A*@axn@$bO zO-Y4JzS&EAi%YG}N?cn?YFS7ivPY=EMV6~YH;+Xxu|tefLS|Aza)Cg6us#)=JW!uH zQa?H>d^j+YHCtyjL^LulF*05|F$RG!AX_OHVI&MtA~_@=5_lU|0000rbW%=J06GH4 z^5LD8b8apw8vNh1ua1mF{{Hy)_U`NA;Nacc+sCpuHXa-V{r&yz?c(9#+}oX+NmiRW z+W-IqK1oDDR5;6GfCDCOP5}iL5fK(cB~ET81`MFgF2kGa9AjhSIk~-E-4&*tPPKdiilQJ11k_J082ZS z>@TvivP!5ZFG?t@{t+GpR3XR&@*hA_VE1|Lo8@L@)l*h(Z@=?c-NS$Fk&&61IzUU9 z*nPqBM=OBZ-6ka1SJgGAS-Us5EN)r#dUX%>wQZLa2ytPCtMKp)Ob z*xcu38Z&d5<-NBS)@jRD+*!W*cf-m_wmxDEqBf?czI%3U0J$Xik;lA`jg}VH?(S(V zE!M3;X2B8w0TnnW&6(8;_Uc)WD;Ms6PKP+s(sFgO!}B!^ES~GDt4qLPxwYB)^7)XA zZwo9zDy-B0B+jT6V=!=bo(zs_8{eBA78gT9GH$(DVhz;4VAYwz+bOIdZ-PNb|I&rl z^XG=vFLF)1{&nT2*0vMz#}7^9hXzzf&ZdKlEj{LihP;|;Ywqn35ajP?H?7t|i-Un% z&&kxee@9B{nwgv1+S-~0)E1{ob1^Wn`F2isurqThKK=3%&;`@{0{!D- z&CSj80t;uPu&FaJFtSXKH#ajgGj}=sEad7US6jP0|Db@0j)?(5@sf<7`~a9>s;wCa zm^)spe{uxGFmrJYI9cOh7s$>8Npkt-5EWB1UKc`{W{y5Ce$1+nM9Cr;);=Ju#N^62OSlJMn7omiUgP&ErsYzT~iGxcW aE(`!K@+CXylaC4j0000 + + + + Coverage report + + + + + +
+
+

Coverage report: + 78% +

+ +
+ +
+ + +
+
+

+ Files + Functions + Classes +

+

+ coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+
+
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Filefunctionstatementsmissingexcludedcoverage
praktikum\__init__.py(no function)000100%
praktikum\bun.pyBun.__init__200100%
praktikum\bun.pyBun.get_name100100%
praktikum\bun.pyBun.get_price100100%
praktikum\bun.py(no function)400100%
praktikum\burger.pyBurger.__init__200100%
praktikum\burger.pyBurger.set_buns100100%
praktikum\burger.pyBurger.add_ingredient100100%
praktikum\burger.pyBurger.remove_ingredient100100%
praktikum\burger.pyBurger.move_ingredient100100%
praktikum\burger.pyBurger.get_price400100%
praktikum\burger.pyBurger.get_receipt600100%
praktikum\burger.py(no function)1100100%
praktikum\database.pyDatabase.__init__1100100%
praktikum\database.pyDatabase.available_buns100100%
praktikum\database.pyDatabase.available_ingredients100100%
praktikum\database.py(no function)800100%
praktikum\ingredient.pyIngredient.__init__300100%
praktikum\ingredient.pyIngredient.get_price100100%
praktikum\ingredient.pyIngredient.get_name100100%
praktikum\ingredient.pyIngredient.get_type100100%
praktikum\ingredient.py(no function)500100%
praktikum\ingredient_types.py(no function)200100%
praktikum\praktikum.pymain121200%
praktikum\praktikum.py(no function)8800%
Total 8920078%
+

+ No items found using the specified filter. +

+
+ + + diff --git a/htmlcov/index.html b/htmlcov/index.html new file mode 100644 index 000000000..a25a4ebef --- /dev/null +++ b/htmlcov/index.html @@ -0,0 +1,153 @@ + + + + + Coverage report + + + + + +
+
+

Coverage report: + 78% +

+ +
+ +
+ + +
+
+

+ Files + Functions + Classes +

+

+ coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+
+
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Filestatementsmissingexcludedcoverage
praktikum\__init__.py000100%
praktikum\bun.py800100%
praktikum\burger.py2700100%
praktikum\database.py2100100%
praktikum\ingredient.py1100100%
praktikum\ingredient_types.py200100%
praktikum\praktikum.py202000%
Total8920078%
+

+ No items found using the specified filter. +

+
+ + + diff --git a/htmlcov/keybd_closed_cb_ce680311.png b/htmlcov/keybd_closed_cb_ce680311.png new file mode 100644 index 0000000000000000000000000000000000000000..ba119c47df81ed2bbd27a06988abf700139c4f99 GIT binary patch literal 9004 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 literal 0 HcmV?d00001 diff --git a/htmlcov/status.json b/htmlcov/status.json new file mode 100644 index 000000000..5530692d3 --- /dev/null +++ b/htmlcov/status.json @@ -0,0 +1 @@ +{"note":"This file is an internal implementation detail to speed up HTML report generation. Its format can change at any time. You might be looking for the JSON report: https://coverage.rtfd.io/cmd.html#cmd-json","format":5,"version":"7.9.2","globals":"a58be93edf661cf82cabb7b98b7544a5","files":{"z_c68eb0c7512457e4___init___py":{"hash":"e6baa73cda2916dad605215f937a92e1","index":{"url":"z_c68eb0c7512457e4___init___py.html","file":"praktikum\\__init__.py","description":"","nums":{"precision":0,"n_files":1,"n_statements":0,"n_excluded":0,"n_missing":0,"n_branches":0,"n_partial_branches":0,"n_missing_branches":0}}},"z_c68eb0c7512457e4_bun_py":{"hash":"820fa7fb820fb46cf300dac8d31b6bd0","index":{"url":"z_c68eb0c7512457e4_bun_py.html","file":"praktikum\\bun.py","description":"","nums":{"precision":0,"n_files":1,"n_statements":8,"n_excluded":0,"n_missing":0,"n_branches":0,"n_partial_branches":0,"n_missing_branches":0}}},"z_c68eb0c7512457e4_burger_py":{"hash":"facdb0a5be4ca4a1f2f321d5a2d8fb87","index":{"url":"z_c68eb0c7512457e4_burger_py.html","file":"praktikum\\burger.py","description":"","nums":{"precision":0,"n_files":1,"n_statements":27,"n_excluded":0,"n_missing":0,"n_branches":0,"n_partial_branches":0,"n_missing_branches":0}}},"z_c68eb0c7512457e4_database_py":{"hash":"30732bb755a2f38ad46148435b66601a","index":{"url":"z_c68eb0c7512457e4_database_py.html","file":"praktikum\\database.py","description":"","nums":{"precision":0,"n_files":1,"n_statements":21,"n_excluded":0,"n_missing":0,"n_branches":0,"n_partial_branches":0,"n_missing_branches":0}}},"z_c68eb0c7512457e4_ingredient_py":{"hash":"d1fcf36fcdce67a9051fb6dc31d4fcd0","index":{"url":"z_c68eb0c7512457e4_ingredient_py.html","file":"praktikum\\ingredient.py","description":"","nums":{"precision":0,"n_files":1,"n_statements":11,"n_excluded":0,"n_missing":0,"n_branches":0,"n_partial_branches":0,"n_missing_branches":0}}},"z_c68eb0c7512457e4_ingredient_types_py":{"hash":"2bcd1d2d9846c4979e7b61455ec13758","index":{"url":"z_c68eb0c7512457e4_ingredient_types_py.html","file":"praktikum\\ingredient_types.py","description":"","nums":{"precision":0,"n_files":1,"n_statements":2,"n_excluded":0,"n_missing":0,"n_branches":0,"n_partial_branches":0,"n_missing_branches":0}}},"z_c68eb0c7512457e4_praktikum_py":{"hash":"a690eb500167123dfff6b25c79079e43","index":{"url":"z_c68eb0c7512457e4_praktikum_py.html","file":"praktikum\\praktikum.py","description":"","nums":{"precision":0,"n_files":1,"n_statements":20,"n_excluded":0,"n_missing":20,"n_branches":0,"n_partial_branches":0,"n_missing_branches":0}}}}} \ No newline at end of file diff --git a/htmlcov/style_cb_db813965.css b/htmlcov/style_cb_db813965.css new file mode 100644 index 000000000..e54e87aed --- /dev/null +++ b/htmlcov/style_cb_db813965.css @@ -0,0 +1,337 @@ +@charset "UTF-8"; +/* Licensed under the Apache License: http://www.apache.org/licenses/LICENSE-2.0 */ +/* For details: https://github.com/nedbat/coveragepy/blob/master/NOTICE.txt */ +/* Don't edit this .css file. Edit the .scss file instead! */ +html, body, h1, h2, h3, p, table, td, th { margin: 0; padding: 0; border: 0; font-weight: inherit; font-style: inherit; font-size: 100%; font-family: inherit; vertical-align: baseline; } + +body { font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; font-size: 1em; background: #fff; color: #000; } + +@media (prefers-color-scheme: dark) { body { background: #1e1e1e; } } + +@media (prefers-color-scheme: dark) { body { color: #eee; } } + +html > body { font-size: 16px; } + +a:active, a:focus { outline: 2px dashed #007acc; } + +p { font-size: .875em; line-height: 1.4em; } + +table { border-collapse: collapse; } + +td { vertical-align: top; } + +table tr.hidden { display: none !important; } + +p#no_rows { display: none; font-size: 1.15em; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; } + +a.nav { text-decoration: none; color: inherit; } + +a.nav:hover { text-decoration: underline; color: inherit; } + +.hidden { display: none; } + +header { background: #f8f8f8; width: 100%; z-index: 2; border-bottom: 1px solid #ccc; } + +@media (prefers-color-scheme: dark) { header { background: black; } } + +@media (prefers-color-scheme: dark) { header { border-color: #333; } } + +header .content { padding: 1rem 3.5rem; } + +header h2 { margin-top: .5em; font-size: 1em; } + +header h2 a.button { font-family: inherit; font-size: inherit; border: 1px solid; border-radius: .2em; background: #eee; color: inherit; text-decoration: none; padding: .1em .5em; margin: 1px calc(.1em + 1px); cursor: pointer; border-color: #ccc; } + +@media (prefers-color-scheme: dark) { header h2 a.button { background: #333; } } + +@media (prefers-color-scheme: dark) { header h2 a.button { border-color: #444; } } + +header h2 a.button.current { border: 2px solid; background: #fff; border-color: #999; cursor: default; } + +@media (prefers-color-scheme: dark) { header h2 a.button.current { background: #1e1e1e; } } + +@media (prefers-color-scheme: dark) { header h2 a.button.current { border-color: #777; } } + +header p.text { margin: .5em 0 -.5em; color: #666; font-style: italic; } + +@media (prefers-color-scheme: dark) { header p.text { color: #aaa; } } + +header.sticky { position: fixed; left: 0; right: 0; height: 2.5em; } + +header.sticky .text { display: none; } + +header.sticky h1, header.sticky h2 { font-size: 1em; margin-top: 0; display: inline-block; } + +header.sticky .content { padding: 0.5rem 3.5rem; } + +header.sticky .content p { font-size: 1em; } + +header.sticky ~ #source { padding-top: 6.5em; } + +main { position: relative; z-index: 1; } + +footer { margin: 1rem 3.5rem; } + +footer .content { padding: 0; color: #666; font-style: italic; } + +@media (prefers-color-scheme: dark) { footer .content { color: #aaa; } } + +#index { margin: 1rem 0 0 3.5rem; } + +h1 { font-size: 1.25em; display: inline-block; } + +#filter_container { float: right; margin: 0 2em 0 0; line-height: 1.66em; } + +#filter_container #filter { width: 10em; padding: 0.2em 0.5em; border: 2px solid #ccc; background: #fff; color: #000; } + +@media (prefers-color-scheme: dark) { #filter_container #filter { border-color: #444; } } + +@media (prefers-color-scheme: dark) { #filter_container #filter { background: #1e1e1e; } } + +@media (prefers-color-scheme: dark) { #filter_container #filter { color: #eee; } } + +#filter_container #filter:focus { border-color: #007acc; } + +#filter_container :disabled ~ label { color: #ccc; } + +@media (prefers-color-scheme: dark) { #filter_container :disabled ~ label { color: #444; } } + +#filter_container label { font-size: .875em; color: #666; } + +@media (prefers-color-scheme: dark) { #filter_container label { color: #aaa; } } + +header button { font-family: inherit; font-size: inherit; border: 1px solid; border-radius: .2em; background: #eee; color: inherit; text-decoration: none; padding: .1em .5em; margin: 1px calc(.1em + 1px); cursor: pointer; border-color: #ccc; } + +@media (prefers-color-scheme: dark) { header button { background: #333; } } + +@media (prefers-color-scheme: dark) { header button { border-color: #444; } } + +header button:active, header button:focus { outline: 2px dashed #007acc; } + +header button.run { background: #eeffee; } + +@media (prefers-color-scheme: dark) { header button.run { background: #373d29; } } + +header button.run.show_run { background: #dfd; border: 2px solid #00dd00; margin: 0 .1em; } + +@media (prefers-color-scheme: dark) { header button.run.show_run { background: #373d29; } } + +header button.mis { background: #ffeeee; } + +@media (prefers-color-scheme: dark) { header button.mis { background: #4b1818; } } + +header button.mis.show_mis { background: #fdd; border: 2px solid #ff0000; margin: 0 .1em; } + +@media (prefers-color-scheme: dark) { header button.mis.show_mis { background: #4b1818; } } + +header button.exc { background: #f7f7f7; } + +@media (prefers-color-scheme: dark) { header button.exc { background: #333; } } + +header button.exc.show_exc { background: #eee; border: 2px solid #808080; margin: 0 .1em; } + +@media (prefers-color-scheme: dark) { header button.exc.show_exc { background: #333; } } + +header button.par { background: #ffffd5; } + +@media (prefers-color-scheme: dark) { header button.par { background: #650; } } + +header button.par.show_par { background: #ffa; border: 2px solid #bbbb00; margin: 0 .1em; } + +@media (prefers-color-scheme: dark) { header button.par.show_par { background: #650; } } + +#help_panel, #source p .annotate.long { display: none; position: absolute; z-index: 999; background: #ffffcc; border: 1px solid #888; border-radius: .2em; color: #333; padding: .25em .5em; } + +#source p .annotate.long { white-space: normal; float: right; top: 1.75em; right: 1em; height: auto; } + +#help_panel_wrapper { float: right; position: relative; } + +#keyboard_icon { margin: 5px; } + +#help_panel_state { display: none; } + +#help_panel { top: 25px; right: 0; padding: .75em; border: 1px solid #883; color: #333; } + +#help_panel .keyhelp p { margin-top: .75em; } + +#help_panel .legend { font-style: italic; margin-bottom: 1em; } + +.indexfile #help_panel { width: 25em; } + +.pyfile #help_panel { width: 18em; } + +#help_panel_state:checked ~ #help_panel { display: block; } + +kbd { border: 1px solid black; border-color: #888 #333 #333 #888; padding: .1em .35em; font-family: SFMono-Regular, Menlo, Monaco, Consolas, monospace; font-weight: bold; background: #eee; border-radius: 3px; } + +#source { padding: 1em 0 1em 3.5rem; font-family: SFMono-Regular, Menlo, Monaco, Consolas, monospace; } + +#source p { position: relative; white-space: pre; } + +#source p * { box-sizing: border-box; } + +#source p .n { float: left; text-align: right; width: 3.5rem; box-sizing: border-box; margin-left: -3.5rem; padding-right: 1em; color: #999; user-select: none; } + +@media (prefers-color-scheme: dark) { #source p .n { color: #777; } } + +#source p .n.highlight { background: #ffdd00; } + +#source p .n a { scroll-margin-top: 6em; text-decoration: none; color: #999; } + +@media (prefers-color-scheme: dark) { #source p .n a { color: #777; } } + +#source p .n a:hover { text-decoration: underline; color: #999; } + +@media (prefers-color-scheme: dark) { #source p .n a:hover { color: #777; } } + +#source p .t { display: inline-block; width: 100%; box-sizing: border-box; margin-left: -.5em; padding-left: 0.3em; border-left: 0.2em solid #fff; } + +@media (prefers-color-scheme: dark) { #source p .t { border-color: #1e1e1e; } } + +#source p .t:hover { background: #f2f2f2; } + +@media (prefers-color-scheme: dark) { #source p .t:hover { background: #282828; } } + +#source p .t:hover ~ .r .annotate.long { display: block; } + +#source p .t .com { color: #008000; font-style: italic; line-height: 1px; } + +@media (prefers-color-scheme: dark) { #source p .t .com { color: #6a9955; } } + +#source p .t .key { font-weight: bold; line-height: 1px; } + +#source p .t .str, #source p .t .fst { color: #0451a5; } + +@media (prefers-color-scheme: dark) { #source p .t .str, #source p .t .fst { color: #9cdcfe; } } + +#source p.mis .t { border-left: 0.2em solid #ff0000; } + +#source p.mis.show_mis .t { background: #fdd; } + +@media (prefers-color-scheme: dark) { #source p.mis.show_mis .t { background: #4b1818; } } + +#source p.mis.show_mis .t:hover { background: #f2d2d2; } + +@media (prefers-color-scheme: dark) { #source p.mis.show_mis .t:hover { background: #532323; } } + +#source p.run .t { border-left: 0.2em solid #00dd00; } + +#source p.run.show_run .t { background: #dfd; } + +@media (prefers-color-scheme: dark) { #source p.run.show_run .t { background: #373d29; } } + +#source p.run.show_run .t:hover { background: #d2f2d2; } + +@media (prefers-color-scheme: dark) { #source p.run.show_run .t:hover { background: #404633; } } + +#source p.exc .t { border-left: 0.2em solid #808080; } + +#source p.exc.show_exc .t { background: #eee; } + +@media (prefers-color-scheme: dark) { #source p.exc.show_exc .t { background: #333; } } + +#source p.exc.show_exc .t:hover { background: #e2e2e2; } + +@media (prefers-color-scheme: dark) { #source p.exc.show_exc .t:hover { background: #3c3c3c; } } + +#source p.par .t { border-left: 0.2em solid #bbbb00; } + +#source p.par.show_par .t { background: #ffa; } + +@media (prefers-color-scheme: dark) { #source p.par.show_par .t { background: #650; } } + +#source p.par.show_par .t:hover { background: #f2f2a2; } + +@media (prefers-color-scheme: dark) { #source p.par.show_par .t:hover { background: #6d5d0c; } } + +#source p .r { position: absolute; top: 0; right: 2.5em; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; } + +#source p .annotate { font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; color: #666; padding-right: .5em; } + +@media (prefers-color-scheme: dark) { #source p .annotate { color: #ddd; } } + +#source p .annotate.short:hover ~ .long { display: block; } + +#source p .annotate.long { width: 30em; right: 2.5em; } + +#source p input { display: none; } + +#source p input ~ .r label.ctx { cursor: pointer; border-radius: .25em; } + +#source p input ~ .r label.ctx::before { content: "▶ "; } + +#source p input ~ .r label.ctx:hover { background: #e8f4ff; color: #666; } + +@media (prefers-color-scheme: dark) { #source p input ~ .r label.ctx:hover { background: #0f3a42; } } + +@media (prefers-color-scheme: dark) { #source p input ~ .r label.ctx:hover { color: #aaa; } } + +#source p input:checked ~ .r label.ctx { background: #d0e8ff; color: #666; border-radius: .75em .75em 0 0; padding: 0 .5em; margin: -.25em 0; } + +@media (prefers-color-scheme: dark) { #source p input:checked ~ .r label.ctx { background: #056; } } + +@media (prefers-color-scheme: dark) { #source p input:checked ~ .r label.ctx { color: #aaa; } } + +#source p input:checked ~ .r label.ctx::before { content: "▼ "; } + +#source p input:checked ~ .ctxs { padding: .25em .5em; overflow-y: scroll; max-height: 10.5em; } + +#source p label.ctx { color: #999; display: inline-block; padding: 0 .5em; font-size: .8333em; } + +@media (prefers-color-scheme: dark) { #source p label.ctx { color: #777; } } + +#source p .ctxs { display: block; max-height: 0; overflow-y: hidden; transition: all .2s; padding: 0 .5em; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; white-space: nowrap; background: #d0e8ff; border-radius: .25em; margin-right: 1.75em; text-align: right; } + +@media (prefers-color-scheme: dark) { #source p .ctxs { background: #056; } } + +#index { font-family: SFMono-Regular, Menlo, Monaco, Consolas, monospace; font-size: 0.875em; } + +#index table.index { margin-left: -.5em; } + +#index td, #index th { text-align: right; padding: .25em .5em; border-bottom: 1px solid #eee; } + +@media (prefers-color-scheme: dark) { #index td, #index th { border-color: #333; } } + +#index td.name, #index th.name { text-align: left; width: auto; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; min-width: 15em; } + +#index th { font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; font-style: italic; color: #333; cursor: pointer; } + +@media (prefers-color-scheme: dark) { #index th { color: #ddd; } } + +#index th:hover { background: #eee; } + +@media (prefers-color-scheme: dark) { #index th:hover { background: #333; } } + +#index th .arrows { color: #666; font-size: 85%; font-family: sans-serif; font-style: normal; pointer-events: none; } + +#index th[aria-sort="ascending"], #index th[aria-sort="descending"] { white-space: nowrap; background: #eee; padding-left: .5em; } + +@media (prefers-color-scheme: dark) { #index th[aria-sort="ascending"], #index th[aria-sort="descending"] { background: #333; } } + +#index th[aria-sort="ascending"] .arrows::after { content: " ▲"; } + +#index th[aria-sort="descending"] .arrows::after { content: " ▼"; } + +#index td.name { font-size: 1.15em; } + +#index td.name a { text-decoration: none; color: inherit; } + +#index td.name .no-noun { font-style: italic; } + +#index tr.total td, #index tr.total_dynamic td { font-weight: bold; border-top: 1px solid #ccc; border-bottom: none; } + +#index tr.region:hover { background: #eee; } + +@media (prefers-color-scheme: dark) { #index tr.region:hover { background: #333; } } + +#index tr.region:hover td.name { text-decoration: underline; color: inherit; } + +#scroll_marker { position: fixed; z-index: 3; right: 0; top: 0; width: 16px; height: 100%; background: #fff; border-left: 1px solid #eee; will-change: transform; } + +@media (prefers-color-scheme: dark) { #scroll_marker { background: #1e1e1e; } } + +@media (prefers-color-scheme: dark) { #scroll_marker { border-color: #333; } } + +#scroll_marker .marker { background: #ccc; position: absolute; min-height: 3px; width: 100%; } + +@media (prefers-color-scheme: dark) { #scroll_marker .marker { background: #444; } } diff --git a/htmlcov/z_c68eb0c7512457e4___init___py.html b/htmlcov/z_c68eb0c7512457e4___init___py.html new file mode 100644 index 000000000..f2034749b --- /dev/null +++ b/htmlcov/z_c68eb0c7512457e4___init___py.html @@ -0,0 +1,97 @@ + + + + + Coverage for praktikum\__init__.py: 100% + + + + + +
+
+

+ Coverage for praktikum\__init__.py: + 100% +

+ +

+ 0 statements   + + + +

+

+ « prev     + ^ index     + » next +       + coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+ +
+
+
+
+ + + diff --git a/htmlcov/z_c68eb0c7512457e4_bun_py.html b/htmlcov/z_c68eb0c7512457e4_bun_py.html new file mode 100644 index 000000000..2dcd3a791 --- /dev/null +++ b/htmlcov/z_c68eb0c7512457e4_bun_py.html @@ -0,0 +1,112 @@ + + + + + Coverage for praktikum\bun.py: 100% + + + + + +
+
+

+ Coverage for praktikum\bun.py: + 100% +

+ +

+ 8 statements   + + + +

+

+ « prev     + ^ index     + » next +       + coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+ +
+
+
+

1class Bun: 

+

2 """ 

+

3 Модель булочки для бургера. 

+

4 Булочке можно дать название и назначить цену. 

+

5 """ 

+

6 

+

7 def __init__(self, name: str, price: float): 

+

8 self.name = name 

+

9 self.price = price 

+

10 

+

11 def get_name(self) -> str: 

+

12 return self.name 

+

13 

+

14 def get_price(self) -> float: 

+

15 return self.price 

+
+ + + diff --git a/htmlcov/z_c68eb0c7512457e4_burger_py.html b/htmlcov/z_c68eb0c7512457e4_burger_py.html new file mode 100644 index 000000000..a6f14665c --- /dev/null +++ b/htmlcov/z_c68eb0c7512457e4_burger_py.html @@ -0,0 +1,145 @@ + + + + + Coverage for praktikum\burger.py: 100% + + + + + +
+
+

+ Coverage for praktikum\burger.py: + 100% +

+ +

+ 27 statements   + + + +

+

+ « prev     + ^ index     + » next +       + coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+ +
+
+
+

1from typing import List 

+

2 

+

3from praktikum.bun import Bun 

+

4from praktikum.ingredient import Ingredient 

+

5 

+

6 

+

7class Burger: 

+

8 """ 

+

9 Модель бургера. 

+

10 Бургер состоит из булочек и ингредиентов (начинка или соус). 

+

11 Ингредиенты можно перемещать и удалять. 

+

12 Можно распечать чек с информацией о бургере. 

+

13 """ 

+

14 

+

15 def __init__(self): 

+

16 self.bun = None 

+

17 self.ingredients: List[Ingredient] = [] 

+

18 

+

19 def set_buns(self, bun: Bun): 

+

20 self.bun = bun 

+

21 

+

22 def add_ingredient(self, ingredient: Ingredient): 

+

23 self.ingredients.append(ingredient) 

+

24 

+

25 def remove_ingredient(self, index: int): 

+

26 del self.ingredients[index] 

+

27 

+

28 def move_ingredient(self, index: int, new_index: int): 

+

29 self.ingredients.insert(new_index, self.ingredients.pop(index)) 

+

30 

+

31 def get_price(self) -> float: 

+

32 price = self.bun.get_price() * 2 

+

33 

+

34 for ingredient in self.ingredients: 

+

35 price += ingredient.get_price() 

+

36 

+

37 return price 

+

38 

+

39 def get_receipt(self) -> str: 

+

40 receipt: List[str] = [f'(==== {self.bun.get_name()} ====)'] 

+

41 

+

42 for ingredient in self.ingredients: 

+

43 receipt.append(f'= {str(ingredient.get_type()).lower()} {ingredient.get_name()} =') 

+

44 

+

45 receipt.append(f'(==== {self.bun.get_name()} ====)\n') 

+

46 receipt.append(f'Price: {self.get_price()}') 

+

47 

+

48 return '\n'.join(receipt) 

+
+ + + diff --git a/htmlcov/z_c68eb0c7512457e4_database_py.html b/htmlcov/z_c68eb0c7512457e4_database_py.html new file mode 100644 index 000000000..9d7abea99 --- /dev/null +++ b/htmlcov/z_c68eb0c7512457e4_database_py.html @@ -0,0 +1,130 @@ + + + + + Coverage for praktikum\database.py: 100% + + + + + +
+
+

+ Coverage for praktikum\database.py: + 100% +

+ +

+ 21 statements   + + + +

+

+ « prev     + ^ index     + » next +       + coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+ +
+
+
+

1from typing import List 

+

2 

+

3from praktikum.bun import Bun 

+

4from praktikum.ingredient import Ingredient 

+

5from praktikum.ingredient_types import INGREDIENT_TYPE_SAUCE, INGREDIENT_TYPE_FILLING 

+

6 

+

7 

+

8class Database: 

+

9 """ 

+

10 Класс с методами по работе с базой данных. 

+

11 """ 

+

12 

+

13 def __init__(self): 

+

14 self.buns: List[Bun] = [] 

+

15 self.ingredients: List[Ingredient] = [] 

+

16 

+

17 self.buns.append(Bun("black bun", 100)) 

+

18 self.buns.append(Bun("white bun", 200)) 

+

19 self.buns.append(Bun("red bun", 300)) 

+

20 

+

21 self.ingredients.append(Ingredient(INGREDIENT_TYPE_SAUCE, "hot sauce", 100)) 

+

22 self.ingredients.append(Ingredient(INGREDIENT_TYPE_SAUCE, "sour cream", 200)) 

+

23 self.ingredients.append(Ingredient(INGREDIENT_TYPE_SAUCE, "chili sauce", 300)) 

+

24 

+

25 self.ingredients.append(Ingredient(INGREDIENT_TYPE_FILLING, "cutlet", 100)) 

+

26 self.ingredients.append(Ingredient(INGREDIENT_TYPE_FILLING, "dinosaur", 200)) 

+

27 self.ingredients.append(Ingredient(INGREDIENT_TYPE_FILLING, "sausage", 300)) 

+

28 

+

29 def available_buns(self) -> List[Bun]: 

+

30 return self.buns 

+

31 

+

32 def available_ingredients(self) -> List[Ingredient]: 

+

33 return self.ingredients 

+
+ + + diff --git a/htmlcov/z_c68eb0c7512457e4_ingredient_py.html b/htmlcov/z_c68eb0c7512457e4_ingredient_py.html new file mode 100644 index 000000000..3674e9c6c --- /dev/null +++ b/htmlcov/z_c68eb0c7512457e4_ingredient_py.html @@ -0,0 +1,117 @@ + + + + + Coverage for praktikum\ingredient.py: 100% + + + + + +
+
+

+ Coverage for praktikum\ingredient.py: + 100% +

+ +

+ 11 statements   + + + +

+

+ « prev     + ^ index     + » next +       + coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+ +
+
+
+

1class Ingredient: 

+

2 """ 

+

3 Модель ингредиента. 

+

4 Ингредиент: начинка или соус. 

+

5 У ингредиента есть тип (начинка или соус), название и цена. 

+

6 """ 

+

7 

+

8 def __init__(self, ingredient_type: str, name: str, price: float): 

+

9 self.type = ingredient_type 

+

10 self.name = name 

+

11 self.price = price 

+

12 

+

13 def get_price(self) -> float: 

+

14 return self.price 

+

15 

+

16 def get_name(self) -> str: 

+

17 return self.name 

+

18 

+

19 def get_type(self) -> str: 

+

20 return self.type 

+
+ + + diff --git a/htmlcov/z_c68eb0c7512457e4_ingredient_types_py.html b/htmlcov/z_c68eb0c7512457e4_ingredient_types_py.html new file mode 100644 index 000000000..5673af4aa --- /dev/null +++ b/htmlcov/z_c68eb0c7512457e4_ingredient_types_py.html @@ -0,0 +1,104 @@ + + + + + Coverage for praktikum\ingredient_types.py: 100% + + + + + +
+
+

+ Coverage for praktikum\ingredient_types.py: + 100% +

+ +

+ 2 statements   + + + +

+

+ « prev     + ^ index     + » next +       + coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+ +
+
+
+

1""" 

+

2Перечисление с типами ингредиентов. 

+

3SAUCE – соус 

+

4FILLING – начинка 

+

5""" 

+

6INGREDIENT_TYPE_SAUCE = 'SAUCE' 

+

7INGREDIENT_TYPE_FILLING = 'FILLING' 

+
+ + + diff --git a/htmlcov/z_c68eb0c7512457e4_praktikum_py.html b/htmlcov/z_c68eb0c7512457e4_praktikum_py.html new file mode 100644 index 000000000..9e2e9c1f0 --- /dev/null +++ b/htmlcov/z_c68eb0c7512457e4_praktikum_py.html @@ -0,0 +1,138 @@ + + + + + Coverage for praktikum\praktikum.py: 0% + + + + + +
+
+

+ Coverage for praktikum\praktikum.py: + 0% +

+ +

+ 20 statements   + + + +

+

+ « prev     + ^ index     + » next +       + coverage.py v7.9.2, + created at 2025-11-14 11:12 +0300 +

+ +
+
+
+

1from typing import List 

+

2 

+

3from praktikum.bun import Bun 

+

4from praktikum.burger import Burger 

+

5from praktikum.database import Database 

+

6from praktikum.ingredient import Ingredient 

+

7 

+

8 

+

9def main(): 

+

10 # Инициализируем базу данных 

+

11 database: Database = Database() 

+

12 

+

13 # Создадим новый бургер 

+

14 burger: Burger = Burger() 

+

15 

+

16 # Считаем список доступных булок из базы данных 

+

17 buns: List[Bun] = database.available_buns() 

+

18 

+

19 # Считаем список доступных ингредиентов из базы данных 

+

20 ingredients: List[Ingredient] = database.available_ingredients() 

+

21 

+

22 # Соберём бургер 

+

23 burger.set_buns(buns[0]) 

+

24 

+

25 burger.add_ingredient(ingredients[1]) 

+

26 burger.add_ingredient(ingredients[4]) 

+

27 burger.add_ingredient(ingredients[3]) 

+

28 burger.add_ingredient(ingredients[5]) 

+

29 

+

30 # Переместим слой с ингредиентом 

+

31 burger.move_ingredient(2, 1) 

+

32 

+

33 # Удалим ингредиент 

+

34 burger.remove_ingredient(3) 

+

35 

+

36 # Распечатаем рецепт бургера 

+

37 print(burger.get_receipt()) 

+

38 

+

39 

+

40if __name__ == "__main__": 

+

41 main() 

+
+ + +