12 # Common parameters for the 'make' during CI tests
13 MAKEFLAGS: "-j5 --no-keep-going"
15 # GitLab-CI environment
17 # more attempts for more robust
18 GET_SOURCES_ATTEMPTS: "10"
19 ARTIFACT_DOWNLOAD_ATTEMPTS: "10"
21 # We use get_sources.sh script to fetch the submodules and/or re-fetch the repo
22 # if it was corrupted (if submodule update fails this can happen)
24 GIT_SUBMODULE_STRATEGY: none
28 IDF_PATH: "$CI_PROJECT_DIR"
31 APPLY_BOT_FILTER_SCRIPT: "$CI_PROJECT_DIR/tools/ci/apply_bot_filter.py"
32 CHECKOUT_REF_SCRIPT: "$CI_PROJECT_DIR/tools/ci/checkout_project_ref.py"
34 # When 'fetch' strategy is used, Gitlab removes untracked files before checking out
35 # new revision. However if the new revision doesn't include some of the submodules
36 # which were present in the old revision, such submodule directories would not be
37 # removed by the checkout. This extra step ensures that these stale submodules
39 .git_clean_stale_submodules: &git_clean_stale_submodules >
40 find . -name '.git' -not -path './.git' -printf '%P\n'
43 grep -q {} .gitmodules
44 || (echo "Removing {}, has .git directory but not in .gitmodules file"
47 # before each job, we need to check if this job is filtered by bot stage/job filter
48 .apply_bot_filter: &apply_bot_filter
49 python $APPLY_BOT_FILTER_SCRIPT || exit 0
52 - *git_clean_stale_submodules
53 # apply bot filter in before script
58 - echo -n $GITLAB_KEY > ~/.ssh/id_rsa_base64
59 - base64 --decode --ignore-garbage ~/.ssh/id_rsa_base64 > ~/.ssh/id_rsa
60 - chmod 600 ~/.ssh/id_rsa
61 - echo -e "Host gitlab.espressif.cn\n\tStrictHostKeyChecking no\n" >> ~/.ssh/config
63 # Set IS_PRIVATE or IS_PUBLIC depending on if our branch is public or not
65 # (the same regular expressions are used to set these are used in 'only:' sections below
66 - source tools/ci/configure_ci_environment.sh
68 # fetch the submodules (& if necessary re-fetch repo) from gitlab
69 - time ./tools/ci/get-full-sources.sh
72 before_script: &do_nothing_before
73 - *git_clean_stale_submodules
74 # apply bot filter in before script
76 - echo "Not setting up GitLab key, not fetching submodules"
77 - source tools/ci/configure_ci_environment.sh
79 .add_gitlab_key_before:
80 before_script: &add_gitlab_key_before
81 - *git_clean_stale_submodules
82 # apply bot filter in before script
84 - echo "Not fetching submodules"
85 - source tools/ci/configure_ci_environment.sh
89 - echo -n $GITLAB_KEY > ~/.ssh/id_rsa_base64
90 - base64 --decode --ignore-garbage ~/.ssh/id_rsa_base64 > ~/.ssh/id_rsa
91 - chmod 600 ~/.ssh/id_rsa
92 - echo -e "Host gitlab.espressif.cn\n\tStrictHostKeyChecking no\n" >> ~/.ssh/config
96 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
103 - git clone https://github.com/espressif/esp-idf-template.git
104 - cd esp-idf-template
105 # Try to use the same branch name for esp-idf-template that we're
106 # using on esp-idf. If it doesn't exist then just stick to the default
108 - python $CHECKOUT_REF_SCRIPT esp-idf-template
109 # Test debug build (default)
111 # Now test release build
113 - sed -i.bak -e's/CONFIG_OPTIMIZATION_LEVEL_DEBUG\=y/CONFIG_OPTIMIZATION_LEVEL_RELEASE=y/' sdkconfig
115 # Check if there are any stray printf/ets_printf references in WiFi libs
116 - cd ../components/esp32/lib
117 - test $(xtensa-esp32-elf-nm *.a | grep -w printf | wc -l) -eq 0
118 - test $(xtensa-esp32-elf-nm *.a | grep -w ets_printf | wc -l) -eq 0
121 .build_template: &build_template
123 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
130 .build_ssc_template: &build_ssc_template
137 SSC_CONFIG_FOLDER: "$CI_PROJECT_DIR/SSC/configs/ESP32_IDF"
139 - git clone $SSC_REPOSITORY
141 - python $CHECKOUT_REF_SCRIPT SSC
142 - MAKEFLAGS= ./ci_build_ssc.sh "${CI_JOB_NAME}" "${IDF_PATH}/.gitlab-ci.yml"
144 # don't forget to add to dependency to test_template when adding new build_ssc jobs
146 <<: *build_ssc_template
149 <<: *build_ssc_template
152 <<: *build_ssc_template
154 # If you want to add new build ssc jobs, please add it into dependencies of `assign_test` and `.test_template`
160 - tools/unit-test-app/output
161 - components/idf_test/unit_test/TestCaseAll.yml
162 - components/idf_test/unit_test/CIConfigs/*.yml
165 - cd tools/unit-test-app
166 - MAKEFLAGS= make help # make sure kconfig tools are built in single process
167 - make ut-clean-all-configs
168 - export EXTRA_CFLAGS="-Werror -Werror=deprecated-declarations"
169 - export EXTRA_CXXFLAGS=${EXTRA_CFLAGS}
170 - make ut-build-all-configs
171 - python tools/UnitTestParser.py
173 .build_examples_template: &build_examples_template
179 - build_examples/*/*/*/build/*.bin
180 - build_examples/*/*/*/build/*.elf
181 - build_examples/*/*/*/build/*.map
182 - build_examples/*/*/*/build/download.config
183 - build_examples/*/*/*/build/bootloader/*.bin
188 LOG_PATH: "$CI_PROJECT_DIR/log_examples"
190 # it's not possible to build 100% out-of-tree and have the "artifacts"
191 # mechanism work, but this is the next best thing
192 - rm -rf build_examples
193 - mkdir build_examples
195 # build some of examples
196 - mkdir -p ${LOG_PATH}
197 - ${IDF_PATH}/tools/ci/build_examples.sh "${CI_JOB_NAME}"
198 # and again, with cmake!
199 - rm -rf ../build_examples_cmake
200 - mkdir ../build_examples_cmake
201 - cd ../build_examples_cmake
202 - mkdir -p ${LOG_PATH}_cmake
203 - LOG_PATH=${LOG_PATH}_cmake ${IDF_PATH}/tools/ci/build_examples_cmake.sh "${CI_JOB_NAME}"
206 <<: *build_examples_template
209 <<: *build_examples_template
212 <<: *build_examples_template
215 <<: *build_examples_template
218 <<: *build_examples_template
221 <<: *build_examples_template
224 <<: *build_examples_template
227 <<: *build_examples_template
229 # If you want to add new build example jobs, please add it into dependencies of `.example_test_template`
233 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
239 # English version of documentation
240 - docs/en/doxygen-warning-log.txt
241 - docs/en/sphinx-warning-log.txt
242 - docs/en/sphinx-warning-log-sanitized.txt
243 - docs/en/_build/html
244 # Chinese version of documentation
245 - docs/zh_CN/doxygen-warning-log.txt
246 - docs/zh_CN/sphinx-warning-log.txt
247 - docs/zh_CN/sphinx-warning-log-sanitized.txt
248 - docs/zh_CN/_build/html
252 - ./check_lang_folder_sync.sh
256 - ../check_doc_warnings.sh
260 - ../check_doc_warnings.sh
264 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
266 tools/cmake/run_cmake_lint.sh
268 .host_test_template: &host_test_template
270 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
276 <<: *host_test_template
278 - cd components/nvs_flash/test_nvs_host
282 <<: *host_test_template
285 - components/nvs_flash/test_nvs_host/coverage_report
288 # This job takes a few hours to finish, so only run it on demand
290 BOT_NEEDS_TRIGGER_BY_NAME: 1
292 - cd components/nvs_flash/test_nvs_host
293 - make coverage_report
295 test_partition_table_on_host:
296 <<: *host_test_template
300 - cd components/partition_table/test_gen_esp32part_host
301 - ./gen_esp32part_tests.py
304 <<: *host_test_template
307 - components/wear_levelling/test_wl_host/coverage_report.zip
309 - cd components/wear_levelling/test_wl_host
313 <<: *host_test_template
315 - cd components/fatfs/test_fatfs_host/
318 .host_fuzzer_test_template: &host_fuzzer_test_template
320 image: $CI_DOCKER_REGISTRY/afl-fuzzer-test
327 - ${FUZZER_TEST_DIR}/out/crashes
328 - ${FUZZER_TEST_DIR}/fuzz_output.log
331 # can only be triggered
334 - export AFL_I_DONT_CARE_ABOUT_MISSING_CRASHES=1 && export AFL_SKIP_CPUFREQ=1
335 - cd ${FUZZER_TEST_DIR}
336 # run AFL fuzzer for one hour
337 - ( ( make ${FUZZER_PARAMS} fuzz | tee fuzz_output.log | grep -v '\(Fuzzing test case\|Entering queue cycle\)' ) || pkill sleep ) &
338 - ( sleep 3600 || mkdir -p out/crashes/env_failed ) && pkill afl-fuz
339 # check no crashes found
340 - test -z "$(ls out/crashes/)" || exit 1
342 test_mdns_fuzzer_on_host:
343 <<: *host_fuzzer_test_template
345 BOT_NEEDS_TRIGGER_BY_NAME: 1
346 FUZZER_TEST_DIR: components/mdns/test_afl_fuzz_host
348 test_lwip_dns_fuzzer_on_host:
349 <<: *host_fuzzer_test_template
351 BOT_NEEDS_TRIGGER_BY_NAME: 1
352 FUZZER_TEST_DIR: components/lwip/test_afl_host
353 FUZZER_PARAMS: MODE=dns
355 test_lwip_dhcp_fuzzer_on_host:
356 <<: *host_fuzzer_test_template
358 BOT_NEEDS_TRIGGER_BY_NAME: 1
359 FUZZER_TEST_DIR: components/lwip/test_afl_host
360 FUZZER_PARAMS: MODE=dhcp_client
362 test_lwip_dhcps_fuzzer_on_host:
363 <<: *host_fuzzer_test_template
365 BOT_NEEDS_TRIGGER_BY_NAME: 1
366 FUZZER_TEST_DIR: components/lwip/test_afl_host
367 FUZZER_PARAMS: MODE=dhcp_server
370 <<: *host_test_template
372 - cd components/spiffs/test_spiffs_host/
375 test_multi_heap_on_host:
376 <<: *host_test_template
378 - cd components/heap/test_multi_heap_host
379 - ./test_all_configs.sh
382 <<: *host_test_template
384 - cd tools/kconfig_new/test
385 - ./test_confserver.py
388 <<: *host_test_template
390 - ${IDF_PATH}/tools/ci/test_configure_ci_environment.sh
391 - rm -rf test_build_system
392 - mkdir test_build_system
393 - cd test_build_system
394 - ${IDF_PATH}/tools/ci/test_build_system.sh
396 test_build_system_cmake:
397 <<: *host_test_template
399 - ${IDF_PATH}/tools/ci/test_configure_ci_environment.sh
400 - rm -rf test_build_system
401 - mkdir test_build_system
402 - cd test_build_system
403 - ${IDF_PATH}/tools/ci/test_build_system_cmake.sh
406 <<: *host_test_template
410 - tools/test_idf_monitor/outputs/*
413 - cd ${IDF_PATH}/tools/test_idf_monitor
414 - source /opt/pyenv/activate
415 - pyenv global 2.7.15
416 - ./run_test_idf_monitor.py
418 - ./run_test_idf_monitor.py
419 - pyenv global system
421 test_esp_err_to_name_on_host:
422 <<: *host_test_template
426 - components/esp32/esp_err_to_name.c
430 - source /opt/pyenv/activate
431 - pyenv global 2.7.15
432 - ./gen_esp_err_to_name.py
433 - git diff --exit-code -- ../components/esp32/esp_err_to_name.c || (echo 'Differences found. Please run gen_esp_err_to_name.py and commit the changes.'; exit 1)
435 - ./gen_esp_err_to_name.py
436 - git diff --exit-code -- ../components/esp32/esp_err_to_name.c || (echo 'Differences found between running under Python 2 and 3.'; exit 1)
437 - pyenv global system
441 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
447 - /^v\d+\.\d+(\.\d+)?($|-)/
450 before_script: *do_nothing_before
454 - echo -n $GH_PUSH_KEY > ~/.ssh/id_rsa_base64
455 - base64 --decode --ignore-garbage ~/.ssh/id_rsa_base64 > ~/.ssh/id_rsa
456 - chmod 600 ~/.ssh/id_rsa
457 - echo -e "Host github.com\n\tStrictHostKeyChecking no\n" >> ~/.ssh/config
458 - git remote remove github &>/dev/null || true
459 - git remote add github git@github.com:espressif/esp-idf.git
460 # Need separate push commands for tag builds and for branch builds
461 - "[ -n \"${CI_COMMIT_TAG}\" ] && git push github ${CI_COMMIT_TAG}"
462 - "[ -z \"${CI_COMMIT_TAG}\" ] && git push github ${CI_COMMIT_SHA}:refs/heads/${CI_COMMIT_REF_NAME}"
466 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
472 - /^v\d+\.\d+(\.\d+)?($|-)/
476 before_script: *do_nothing_before
480 - echo -n $DOCS_DEPLOY_KEY > ~/.ssh/id_rsa_base64
481 - base64 --decode --ignore-garbage ~/.ssh/id_rsa_base64 > ~/.ssh/id_rsa
482 - chmod 600 ~/.ssh/id_rsa
483 - echo -e "Host $DOCS_SERVER\n\tStrictHostKeyChecking no\n\tUser $DOCS_SERVER_USER\n" >> ~/.ssh/config
484 - export GIT_VER=$(git describe --always)
487 - tar czvf $GIT_VER.tar.gz $GIT_VER
488 - scp $GIT_VER.tar.gz $DOCS_SERVER:$DOCS_PATH/en
489 - ssh $DOCS_SERVER -x "cd $DOCS_PATH/en && tar xzvf $GIT_VER.tar.gz && rm -f latest && ln -s $GIT_VER latest"
490 - cd ../../zh_CN/_build/
492 - tar czvf $GIT_VER.tar.gz $GIT_VER
493 - scp $GIT_VER.tar.gz $DOCS_SERVER:$DOCS_PATH/zh_CN
494 - ssh $DOCS_SERVER -x "cd $DOCS_PATH/zh_CN && tar xzvf $GIT_VER.tar.gz && rm -f latest && ln -s $GIT_VER latest"
495 # add link to preview doc
496 - echo "[document preview][en] $CI_DOCKER_REGISTRY/docs/esp-idf/en/${GIT_VER}/index.html"
497 - echo "[document preview][zh_CN] $CI_DOCKER_REGISTRY/docs/esp-idf/zh_CN/${GIT_VER}/index.html"
501 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
505 # can only be triggered
509 - docs/_build/linkcheck
512 # must be triggered with CHECK_LINKS=Yes, otherwise exit without test
513 - test "$CHECK_LINKS" = "Yes" || exit 0
514 # can only run on master branch (otherwise the commit is not on Github yet)
515 - test "${CI_COMMIT_REF_NAME}" = "master" || exit 0
521 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
527 - /^v\d+\.\d+(\.\d+)?($|-)/
529 before_script: *do_nothing_before
531 - tools/ci/check-line-endings.sh ${IDF_PATH}
535 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
541 - /^v\d+\.\d+(\.\d+)?($|-)/
543 before_script: *do_nothing_before
546 - git log -n10 --oneline
547 # commit start with "WIP: " need to be squashed before merge
548 - 'git log --pretty=%s master.. -- | grep "^WIP: " && exit 1 || exit 0'
552 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
558 - /^v\d+\.\d+(\.\d+)?($|-)/
560 before_script: *do_nothing_before
562 - tools/ci/check-executable.sh
564 check_examples_cmake_make:
566 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
572 - /^v\d+\.\d+(\.\d+)?($|-)/
574 before_script: *do_nothing_before
576 - tools/ci/check_examples_cmake_make.sh
578 check_submodule_sync:
580 image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG
586 - /^v\d+\.\d+(\.\d+)?($|-)/
590 before_script: *do_nothing_before
592 # check if all submodules are correctly synced to public repostory
593 - git submodule update --init --recursive
598 image: $CI_DOCKER_REGISTRY/ubuntu-test-env
600 # gitlab ci do not support match job with RegEx or wildcard now in dependencies.
601 # we have a lot build example jobs. now we don't use dependencies, just download all artificats of build stage.
606 - build_esp_idf_tests
608 TEST_FW_PATH: "$CI_PROJECT_DIR/tools/tiny-test-fw"
609 EXAMPLE_CONFIG_OUTPUT_PATH: "$CI_PROJECT_DIR/examples/test_configs"
612 - components/idf_test/*/CIConfigs
613 - components/idf_test/*/TC.sqlite
614 - $EXAMPLE_CONFIG_OUTPUT_PATH
616 before_script: *add_gitlab_key_before
618 # assign example tests
619 - python $TEST_FW_PATH/CIAssignExampleTest.py $IDF_PATH/examples $IDF_PATH/.gitlab-ci.yml $EXAMPLE_CONFIG_OUTPUT_PATH
620 # assign unit test cases
621 - python $TEST_FW_PATH/CIAssignUnitTest.py $IDF_PATH/components/idf_test/unit_test/TestCaseAll.yml $IDF_PATH/.gitlab-ci.yml $IDF_PATH/components/idf_test/unit_test/CIConfigs
622 # clone test script to assign tests
623 - git clone $TEST_SCRIPT_REPOSITORY
624 - cd auto_test_script
625 - python $CHECKOUT_REF_SCRIPT auto_test_script
626 # assgin integration test cases
627 - python CIAssignTestCases.py -t $IDF_PATH/components/idf_test/integration_test -c $IDF_PATH/.gitlab-ci.yml -b $IDF_PATH/SSC/ssc_bin
629 .example_test_template: &example_test_template
630 stage: integration_test
635 - /^v\d+\.\d+(\.\d+)?($|-)/
654 TEST_FW_PATH: "$CI_PROJECT_DIR/tools/tiny-test-fw"
655 TEST_CASE_PATH: "$CI_PROJECT_DIR/examples"
656 CONFIG_FILE: "$CI_PROJECT_DIR/examples/test_configs/$CI_JOB_NAME.yml"
657 LOG_PATH: "$CI_PROJECT_DIR/TEST_LOGS"
658 ENV_FILE: "$CI_PROJECT_DIR/ci-test-runner-configs/$CI_RUNNER_DESCRIPTION/EnvConfig.yml"
660 # first test if config file exists, if not exist, exit 0
661 - test -e $CONFIG_FILE || exit 0
662 # clone test env configs
663 - git clone $TEST_ENV_CONFIG_REPOSITORY
664 - cd ci-test-runner-configs
665 - python $CHECKOUT_REF_SCRIPT ci-test-runner-configs
668 - python Runner.py $TEST_CASE_PATH -c $CONFIG_FILE -e $ENV_FILE
670 .unit_test_template: &unit_test_template
671 <<: *example_test_template
675 - build_esp_idf_tests
677 TEST_FW_PATH: "$CI_PROJECT_DIR/tools/tiny-test-fw"
678 TEST_CASE_PATH: "$CI_PROJECT_DIR/tools/unit-test-app"
679 CONFIG_FILE: "$CI_PROJECT_DIR/components/idf_test/unit_test/CIConfigs/$CI_JOB_NAME.yml"
680 LOG_PATH: "$CI_PROJECT_DIR/TEST_LOGS"
681 ENV_FILE: "$CI_PROJECT_DIR/ci-test-runner-configs/$CI_RUNNER_DESCRIPTION/EnvConfig.yml"
683 .test_template: &test_template
684 stage: integration_test
689 - /^v\d+\.\d+(\.\d+)?($|-)/
703 LOCAL_ENV_CONFIG_PATH: "$CI_PROJECT_DIR/ci-test-runner-configs/$CI_RUNNER_DESCRIPTION/ESP32_IDF"
704 LOG_PATH: "$CI_PROJECT_DIR/$CI_COMMIT_SHA"
705 TEST_CASE_FILE_PATH: "$CI_PROJECT_DIR/components/idf_test/integration_test"
706 MODULE_UPDATE_FILE: "$CI_PROJECT_DIR/components/idf_test/ModuleDefinition.yml"
707 CONFIG_FILE: "$CI_PROJECT_DIR/components/idf_test/integration_test/CIConfigs/$CI_JOB_NAME.yml"
708 before_script: *add_gitlab_key_before
710 # first test if config file exists, if not exist, exit 0
711 - test -e $CONFIG_FILE || exit 0
712 # clone local test env configs
713 - git clone $TEST_ENV_CONFIG_REPOSITORY
714 - cd ci-test-runner-configs
715 - python $CHECKOUT_REF_SCRIPT ci-test-runner-configs
717 - git clone $TEST_SCRIPT_REPOSITORY
718 - cd auto_test_script
719 - python $CHECKOUT_REF_SCRIPT auto_test_script
721 - python CIRunner.py -l "$LOG_PATH/$CI_JOB_NAME" -c $CONFIG_FILE -e $LOCAL_ENV_CONFIG_PATH -t $TEST_CASE_FILE_PATH -m $MODULE_UPDATE_FILE
735 # clone local test env configs
736 - git clone $TEST_ENV_CONFIG_REPOSITORY
737 - cd ci-test-runner-configs
738 - python $CHECKOUT_REF_SCRIPT ci-test-runner-configs
740 - git clone $TEST_SCRIPT_REPOSITORY
741 - cd auto_test_script
742 - git checkout ${CI_COMMIT_REF_NAME} || echo "Using default branch..."
744 - ./Tools/prepare_nvs_bin.sh
746 - python CIRunner.py -l "$LOG_PATH/$CI_JOB_NAME" -c $CONFIG_FILE -e $LOCAL_ENV_CONFIG_PATH -t $TEST_CASE_FILE_PATH -m $MODULE_UPDATE_FILE
749 <<: *example_test_template
755 <<: *example_test_template
761 <<: *example_test_template
767 <<: *example_test_template
773 <<: *unit_test_template
779 <<: *unit_test_template
785 <<: *unit_test_template
791 <<: *unit_test_template
797 <<: *unit_test_template
803 <<: *unit_test_template
809 <<: *unit_test_template
815 <<: *unit_test_template
821 <<: *unit_test_template
827 <<: *unit_test_template
833 <<: *unit_test_template
839 <<: *unit_test_template
845 <<: *unit_test_template
851 <<: *unit_test_template
857 <<: *unit_test_template
863 <<: *unit_test_template
869 <<: *unit_test_template
875 <<: *unit_test_template
881 <<: *unit_test_template
887 <<: *unit_test_template
893 <<: *unit_test_template
899 <<: *unit_test_template
905 <<: *unit_test_template
911 <<: *unit_test_template
917 <<: *unit_test_template
923 <<: *unit_test_template
929 <<: *unit_test_template
935 <<: *unit_test_template
941 <<: *unit_test_template
947 <<: *unit_test_template
953 <<: *unit_test_template
959 <<: *unit_test_template
965 <<: *unit_test_template
971 <<: *unit_test_template
977 <<: *unit_test_template
983 <<: *unit_test_template
989 <<: *unit_test_template
995 <<: *unit_test_template
1001 <<: *unit_test_template
1007 <<: *unit_test_template
1013 <<: *unit_test_template
1019 <<: *unit_test_template
1025 <<: *unit_test_template
1032 <<: *unit_test_template
1039 <<: *unit_test_template
1046 <<: *unit_test_template
1053 <<: *unit_test_template
1060 <<: *unit_test_template
1067 <<: *unit_test_template
1074 <<: *unit_test_template
1081 <<: *unit_test_template
1088 <<: *unit_test_template
1095 <<: *unit_test_template
1102 <<: *unit_test_template
1109 <<: *unit_test_template
1116 <<: *unit_test_template
1122 <<: *unit_test_template
1128 <<: *unit_test_template
1134 <<: *unit_test_template
1141 <<: *unit_test_template
1147 <<: *unit_test_template
1153 <<: *unit_test_template
1159 <<: *unit_test_template
1166 <<: *unit_test_template
1172 <<: *unit_test_template
1178 <<: *unit_test_template
1184 <<: *unit_test_template
1191 <<: *unit_test_template
1197 <<: *unit_test_template
1203 <<: *unit_test_template
1209 <<: *unit_test_template
1216 <<: *unit_test_template