From 6200566f80a71098f7117e4e2f9770baea444183 Mon Sep 17 00:00:00 2001 From: Hu Yueh-Wei Date: Mon, 7 Oct 2024 22:58:01 +0800 Subject: [PATCH] doc: refine extension standalone testing (#72) --- .../ten_testing/for_users_of_ten_framework.md | 40 +++++++++++++++++-- 1 file changed, 36 insertions(+), 4 deletions(-) diff --git a/docs/ten_framework/ten_testing/for_users_of_ten_framework.md b/docs/ten_framework/ten_testing/for_users_of_ten_framework.md index 78514f142b..46b08e2a76 100644 --- a/docs/ten_framework/ten_testing/for_users_of_ten_framework.md +++ b/docs/ten_framework/ten_testing/for_users_of_ten_framework.md @@ -14,6 +14,10 @@ The standalone testing framework for TEN extensions follows two key principles: The exact same extension code used in runtime can be fully tested using this standalone testing framework. +3. The standalone testing flow for extensions developed in different languages follows the same design principles and usage methods. + + This ensures that once you learn the standalone testing concepts and methods for one language, you can quickly grasp the standalone testing concepts and methods for TEN extensions in other languages. + For users, the TEN extension standalone testing framework introduces two main concepts: 1. **extension_tester** @@ -21,6 +25,10 @@ For users, the TEN extension standalone testing framework introduces two main co The role of **extension_tester** is similar to a testing driver, responsible for setting up and executing the entire testing process. **ten_env_tester**, on the other hand, behaves like a typical TEN extension's `ten_env` instance, enabling users to invoke functionalities within the standalone testing framework from the callback of `extension_tester`, such as sending messages to the extension under test and receiving messages from the extension. +From the API design of **extension_tester** and **ten_env_tester**, you can see that they are very similar to **TEN extension** and **ten_env**, and this is by design. The main purpose is to allow users familiar with extension development to quickly get accustomed to the standalone testing framework and then develop standalone test cases for the extension itself. + +However, for testing purposes, there are inevitably APIs and features dedicated specifically to testing. To prevent these test-specific functionalities and APIs, which are not needed during actual runtime, from polluting the runtime API set, the standalone testing framework is designed not to directly use or extend the API sets of **extension** and **ten_env**. Instead, it introduces types and API sets that are exclusive to standalone testing. This separation ensures that the types and API sets used in actual runtime and those used during testing do not interfere with each other, avoiding any negative side effects. + ## Standalone Testing Framework Internal Internally, the TEN extension standalone testing framework implicitly starts a test app, loads the extension addon to be tested (let’s call it extension A), and initializes a graph containing both the extension A to be tested and another extension (let’s call it extension B) dedicated to testing other extensions. All input and output messages of extension A are redirected to extension B, allowing users to customize inputs and outputs during the testing process and complete the actual testing work. @@ -31,13 +39,24 @@ sequenceDiagram participant testing as Extension Testing participant tested as Extension Tested - tester->>testing - testing->>tested + tester->>testing: send messages + testing->>tested: send messages - tested-->>testing - testing-->>tester + tested-->>testing: send results + testing-->>tester: send results ``` +Basically, you can think of this testing extension hidden within the testing framework as a proxy between the extension being tested and the tester. It acts as a proxy extension within the TEN graph, facilitating the exchange of messages between the tested extension and the tester using the language of the TEN environment. + +## Basic Testing Process + +The basic testing process and logic are as follows: + +1. Create an extension tester to manage the entire standalone testing process. +2. Inform the standalone testing framework of the folder containing the extension to be tested. +3. Set a testing mode, such as a mode for testing a single extension. +4. Start the testing. + ## C++ Here is an example of TEN extension standalone testing using Google gtest: @@ -58,8 +77,10 @@ class extension_tester_basic : public ten::extension_tester_t { }; TEST(Test, Basic) { + // 1. Create an extension tester to manage the entire standalone testing process. auto *tester = new extension_tester_basic(); + // 2. Inform the standalone testing framework of the folder containing the extension to be tested. ten_string_t *path = ten_path_get_executable_path(); ten_path_join_c_str(path, "../ten_packages/extension/default_extension_cpp/"); @@ -67,8 +88,12 @@ TEST(Test, Basic) { ten_string_destroy(path); + // 3. Set a testing mode, such as a mode for testing a single extension. tester->set_test_mode_single("default_extension_cpp"); + + // 4. Start the testing. tester->run(); + delete tester; } ``` @@ -102,8 +127,15 @@ class ExtensionTesterBasic(ExtensionTester): def test_basic(): + # 1. Create an extension tester to manage the entire standalone testing process. tester = ExtensionTesterBasic() + + # 2. Inform the standalone testing framework of the folder containing the extension to be tested. tester.add_addon_base_dir(str(Path(__file__).resolve().parent.parent)) + + # 3. Set a testing mode, such as a mode for testing a single extension. tester.set_test_mode_single("default_extension_python") + + # 4. Start the testing. tester.run() ```