|
838 | 838 | "metadata": {},
|
839 | 839 | "source": [
|
840 | 840 | "### Relative imports\n",
|
| 841 | + "\n", |
841 | 842 | "- import module, class, and function, relative to the current module\n",
|
842 | 843 | "- use `.` for the current package\n",
|
843 | 844 | "- use `..` for parent package\n",
|
|
848 | 849 | "```"
|
849 | 850 | ]
|
850 | 851 | },
|
| 852 | + { |
| 853 | + "cell_type": "markdown", |
| 854 | + "metadata": {}, |
| 855 | + "source": [ |
| 856 | + "## Unit testing with assert\n", |
| 857 | + "\n", |
| 858 | + "- testing is an essential part of software development\n", |
| 859 | + "- testing helps ensure that the code (functions) work as expected\n", |
| 860 | + "- testing helps catch bugs and errors early in the development cycle\n", |
| 861 | + "- testing helps improve code quality and maintainability\n", |
| 862 | + "- testing helps document the code and its behavior\n", |
| 863 | + "- testing helps ensure that new changes don't break existing functionality\n", |
| 864 | + "- testing helps ensure that the code is robust and reliable\n", |
| 865 | + "- testing helps ensure that the code is secure and performant\n", |
| 866 | + "- if there was no library and tools provided for unit testing, you could write your own\n", |
| 867 | + "- use the `assert` statement to assert how the two values are compared\n", |
| 868 | + " - mostly equal!" |
| 869 | + ] |
| 870 | + }, |
| 871 | + { |
| 872 | + "cell_type": "code", |
| 873 | + "execution_count": 2, |
| 874 | + "metadata": {}, |
| 875 | + "outputs": [], |
| 876 | + "source": [ |
| 877 | + "assert 'hello' == 'hello'" |
| 878 | + ] |
| 879 | + }, |
| 880 | + { |
| 881 | + "cell_type": "code", |
| 882 | + "execution_count": 3, |
| 883 | + "metadata": {}, |
| 884 | + "outputs": [], |
| 885 | + "source": [ |
| 886 | + "assert 1.5 == 1.5000" |
| 887 | + ] |
| 888 | + }, |
| 889 | + { |
| 890 | + "cell_type": "code", |
| 891 | + "execution_count": 4, |
| 892 | + "metadata": {}, |
| 893 | + "outputs": [ |
| 894 | + { |
| 895 | + "ename": "AssertionError", |
| 896 | + "evalue": "", |
| 897 | + "output_type": "error", |
| 898 | + "traceback": [ |
| 899 | + "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m", |
| 900 | + "\u001b[0;31mAssertionError\u001b[0m Traceback (most recent call last)", |
| 901 | + "Cell \u001b[0;32mIn[4], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m \u001b[38;5;28;01massert\u001b[39;00m \u001b[38;5;241m2.5\u001b[39m \u001b[38;5;241m==\u001b[39m \u001b[38;5;241m3\u001b[39m\n", |
| 902 | + "\u001b[0;31mAssertionError\u001b[0m: " |
| 903 | + ] |
| 904 | + } |
| 905 | + ], |
| 906 | + "source": [ |
| 907 | + "assert 2.5 == 3" |
| 908 | + ] |
| 909 | + }, |
| 910 | + { |
| 911 | + "cell_type": "code", |
| 912 | + "execution_count": 5, |
| 913 | + "metadata": {}, |
| 914 | + "outputs": [], |
| 915 | + "source": [ |
| 916 | + "def add(a: int, b: int) -> int:\n", |
| 917 | + " return a + b" |
| 918 | + ] |
| 919 | + }, |
| 920 | + { |
| 921 | + "cell_type": "code", |
| 922 | + "execution_count": 7, |
| 923 | + "metadata": {}, |
| 924 | + "outputs": [], |
| 925 | + "source": [ |
| 926 | + "def test_add_positives():\n", |
| 927 | + " assert add(1, 2) == 3\n", |
| 928 | + "\n", |
| 929 | + "def test_add_negatives():\n", |
| 930 | + " assert add(-1, -2) == -3\n", |
| 931 | + "\n", |
| 932 | + "def test_add_mixed():\n", |
| 933 | + " assert add(-1, 1) == 0" |
| 934 | + ] |
| 935 | + }, |
| 936 | + { |
| 937 | + "cell_type": "code", |
| 938 | + "execution_count": 8, |
| 939 | + "metadata": {}, |
| 940 | + "outputs": [], |
| 941 | + "source": [ |
| 942 | + "test_add_positives()\n", |
| 943 | + "test_add_negatives()\n", |
| 944 | + "test_add_mixed()" |
| 945 | + ] |
| 946 | + }, |
| 947 | + { |
| 948 | + "cell_type": "markdown", |
| 949 | + "metadata": {}, |
| 950 | + "source": [ |
| 951 | + "## Pytest\n", |
| 952 | + "\n", |
| 953 | + "- Pytest helps your write better programs: [https://docs.pytest.org/en/8.0.x/](https://docs.pytest.org/en/8.0.x/)\n", |
| 954 | + "- Must explictly install pytest; not a part of standard library\n", |
| 955 | + "- `pip install -U pytest`\n", |
| 956 | + "- pytest can be executed for a single file, or all the files and subfolders within a folder\n", |
| 957 | + "- pytest can discover all the test modules that are named `test_*.py`\n", |
| 958 | + "- pytest will execute every `test_*()` function in the `test` module" |
| 959 | + ] |
| 960 | + }, |
| 961 | + { |
| 962 | + "cell_type": "markdown", |
| 963 | + "metadata": {}, |
| 964 | + "source": [ |
| 965 | + "## Code coverage by unit tests\n", |
| 966 | + "\n", |
| 967 | + "- though not sufficient, code coverage provides measureable metrics to determine how good the test cases are\n", |
| 968 | + "- we can use `pytest-cov` plugin for pytest to demonstrate the code coverage from our tests\n", |
| 969 | + "- it must be installed `pip install pytest-cov`\n", |
| 970 | + "- more on it here: [https://pytest-cov.readthedocs.io/en/latest/readme.html](https://pytest-cov.readthedocs.io/en/latest/readme.html)\n", |
| 971 | + "- see [https://github.com/rambasnet/course-container](https://github.com/rambasnet/course-container)\n", |
| 972 | + " - uses [https://app.codecov.io/](https://app.codecov.io/) that provides code coverage report and badge if ci-cd is configured\n", |
| 973 | + " - you can login with github on the service and just configure each repo to use the app\n", |
| 974 | + " - see the ci-test.yml file: [https://github.com/rambasnet/course-container/blob/main/.github/workflows/ci-test.yml](https://github.com/rambasnet/course-container/blob/main/.github/workflows/ci-test.yml)\n", |
| 975 | + "- see and run the Makefile in the root folder of this repo from a terminal" |
| 976 | + ] |
| 977 | + }, |
851 | 978 | {
|
852 | 979 | "cell_type": "markdown",
|
853 | 980 | "metadata": {},
|
|
0 commit comments