:orphan: .. index:: pair: page; Job TAPI usage scenarios .. _doxid-group__tapi__job_tapi-job-scenarios: Job TAPI usage scenarios ======================== Run and wait ~~~~~~~~~~~~ * Run a program * Log all stdout and stderr via the Logger * Wait for the program termination .. ref-code-block:: cpp :ref:`tapi_job_factory_t ` *factory = NULL; :ref:`tapi_job_t ` *job = NULL :ref:`tapi_job_channel_t ` *out_channels[2]; :ref:`CHECK_RC `(:ref:`tapi_job_factory_rpc_create `(pco, &factory)); :ref:`CHECK_RC `(:ref:`tapi_job_create `(factory, NULL, "/usr/bin/tool", (const char *[]){"tool", "arg1", "arg2", NULL}, NULL, &job)); :ref:`CHECK_RC `(:ref:`tapi_job_alloc_output_channels `(job, 2, out_channels)); :ref:`CHECK_RC `(:ref:`tapi_job_attach_filter `(:ref:`TAPI_JOB_CHANNEL_SET `(out_channels[0], out_channels[1]), "Filter", false, :ref:`TE_LL_INFO `, NULL)); :ref:`CHECK_RC `(:ref:`tapi_job_start `(job)); ... CHECK_RC(:ref:`tapi_job_wait `(job, -1, &status)); if (status.type != :ref:`TAPI_JOB_STATUS_EXITED ` || status.value != 0) :ref:`TEST_FAIL `("Tool failed"); :ref:`CHECK_RC `(:ref:`tapi_job_destroy `(job, -1)); :ref:`tapi_job_factory_destroy `(factory); Run and read ~~~~~~~~~~~~ * Run a program * Read all the stdout while logging all the stderr * When all the data are read, the program is assumed to terminate .. ref-code-block:: cpp :ref:`tapi_job_factory_t ` *factory = NULL; :ref:`tapi_job_t ` *job = NULL; :ref:`tapi_job_buffer_t ` buf = :ref:`TAPI_JOB_BUFFER_INIT `; :ref:`tapi_job_channel_t ` *out_channels[2]; :ref:`tapi_job_channel_t ` *out_filter; :ref:`CHECK_RC `(:ref:`tapi_job_factory_rpc_create `(pco, &factory)); :ref:`CHECK_RC `(:ref:`tapi_job_create `(factory, NULL, "/usr/bin/tool", (const char *[]){"tool", "arg1", "arg2", NULL}, NULL, &job)); :ref:`CHECK_RC `(:ref:`tapi_job_alloc_output_channels `(job, 2, out_channels)); :ref:`CHECK_RC `(:ref:`tapi_job_attach_filter `(:ref:`TAPI_JOB_CHANNEL_SET `(out_channels[0]), "Out filter", true, 0, &out_filter)); :ref:`CHECK_RC `(:ref:`tapi_job_attach_filter `(:ref:`TAPI_JOB_CHANNEL_SET `(out_channels[1]), "Error filter", false, :ref:`TE_LL_INFO `, NULL)); :ref:`CHECK_RC `(:ref:`tapi_job_start `(job)); while (!buf.:ref:`eos `) { :ref:`te_string_reset `(&buf.:ref:`data `); :ref:`CHECK_RC `(:ref:`tapi_job_receive `(:ref:`TAPI_JOB_CHANNEL_SET `(out_filter), -1, &buf)); ... } :ref:`CHECK_RC `(:ref:`tapi_job_wait `(job, 0, &status)); if (status.type != :ref:`TAPI_JOB_STATUS_EXITED ` || status.value != 0) :ref:`TEST_FAIL `("Tool failed"); :ref:`CHECK_RC `(:ref:`tapi_job_destroy `(job, -1)); :ref:`tapi_job_factory_destroy `(factory); Run and wait for ~~~~~~~~~~~~~~~~ * Run a program * Log the stderr line matching ``ERROR`` as TE errors * Wait for the string ``Completed`` at the stdout .. ref-code-block:: cpp :ref:`tapi_job_factory_t ` *factory = NULL; :ref:`tapi_job_t ` *job = NULL; :ref:`tapi_job_channel_t ` *out_channels[2]; :ref:`tapi_job_channel_t ` *out_filters[2]; :ref:`CHECK_RC `(:ref:`tapi_job_factory_rpc_create `(pco, &factory)); :ref:`CHECK_RC `(:ref:`tapi_job_create `(factory, NULL, "/usr/bin/tool", (const char *[]){"tool", "arg1", "arg2", NULL}, NULL, &job)); :ref:`CHECK_RC `(:ref:`tapi_job_alloc_output_channels `(job, 2, out_channels)); :ref:`CHECK_RC `(:ref:`tapi_job_attach_filter `(:ref:`TAPI_JOB_CHANNEL_SET `(out_channels[0]), "Out filter", true, 0, &out_filters[0])); :ref:`CHECK_RC `(:ref:`tapi_job_filter_add_regexp `(out_filters[0], "Completed", 0)); :ref:`CHECK_RC `(:ref:`tapi_job_attach_filter `(:ref:`TAPI_JOB_CHANNEL_SET `(out_channels[1]), "Error filter", false, :ref:`TE_LL_ERROR `, &out_filters[1])); :ref:`CHECK_RC `(:ref:`tapi_job_filter_add_regexp `(out_filters[1], "^ERROR:(.*)$", 1); :ref:`CHECK_RC `(:ref:`tapi_job_start `(job)); :ref:`CHECK_RC `(:ref:`tapi_job_poll `(:ref:`TAPI_JOB_CHANNEL_SET `(out_filters[0]), -1)); :ref:`CHECK_RC `(:ref:`tapi_job_wait `(job, 0, &status)); if (status.type != :ref:`TAPI_JOB_STATUS_EXITED ` || status.value != 0) :ref:`TEST_FAIL `("Tool failed"); :ref:`CHECK_RC `(:ref:`tapi_job_destroy `(job, -1)); :ref:`tapi_job_factory_destroy `(factory); Run an interactive program ~~~~~~~~~~~~~~~~~~~~~~~~~~ * Run an interactive program via a pseudo-terminal * Wait for the prompt * Send a command * Repeat the previous steps, logging all the output until the program terminates .. ref-code-block:: cpp :ref:`tapi_job_factory_t ` *factory = NULL; :ref:`tapi_job_t ` *job = NULL; :ref:`tapi_job_channel_t ` *in_channel; :ref:`tapi_job_channel_t ` *out_channel; :ref:`tapi_job_channel_t ` *prompt_filter; :ref:`CHECK_RC `(:ref:`tapi_job_factory_rpc_create `(pco, &factory)); :ref:`CHECK_RC `(:ref:`tapi_job_create `(factory, "pty_spawner", "/usr/bin/tool", (const char *[]){"tool", "arg1", "arg2", NULL}, NULL, &job)); :ref:`CHECK_RC `(:ref:`tapi_job_alloc_output_channels `(job, 1, &out_channel)); :ref:`CHECK_RC `(:ref:`tapi_job_alloc_input_channels `(job, 1, &in_channel)); :ref:`CHECK_RC `(:ref:`tapi_job_attach_filter `(:ref:`TAPI_JOB_CHANNEL_SET `(out_channel), "Readable filter", true, 0, &prompt_filter)); :ref:`CHECK_RC `(:ref:`tapi_job_filter_add_regexp `(prompt_filter, "^\\$ ", 0)); :ref:`CHECK_RC `(:ref:`tapi_job_attach_filter `(:ref:`TAPI_JOB_CHANNEL_SET `(out_channel), "Log filter", false, :ref:`TE_LL_RING `, NULL)); :ref:`CHECK_RC `(:ref:`tapi_job_start `(job)); while (:ref:`tapi_job_wait `(job, 0, &status) == :ref:`TE_EINPROGRESS `) { :ref:`CHECK_RC `(:ref:`tapi_job_receive `(:ref:`TAPI_JOB_CHANNEL_SET `(prompt_filter), -1, NULL)); :ref:`CHECK_RC `(:ref:`tapi_job_send `(job, in_channel, &command_string)); } if (status.type != :ref:`TAPI_JOB_STATUS_EXITED ` || status.value != 0) :ref:`TEST_FAIL `("Tool failed"); :ref:`CHECK_RC `(:ref:`tapi_job_destroy `(job, -1)); :ref:`tapi_job_factory_destroy `(factory); Terminate a job and if it hangs, forcedly kill it ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ .. ref-code-block:: cpp ... CHECK_RC(:ref:`tapi_job_kill `(job, SIGTERM)); if (:ref:`tapi_job_wait `(job, TIMEOUT, &status) == TE_INPROGRESS) { :ref:`CHECK_RC `(:ref:`tapi_job_kill `(job, SIGKILL)); :ref:`CHECK_RC `(:ref:`tapi_job_wait `(job, 0, &status)); } ... Start a job several times and get its data ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ .. ref-code-block:: cpp unsigned int eos_count; tapi_job_buffer buf = :ref:`TAPI_JOB_BUFFER_INIT `; :ref:`tapi_job_factory_t ` *factory = NULL; :ref:`tapi_job_t ` *job = NULL; :ref:`tapi_job_channel_t ` *out_channel; :ref:`tapi_job_channel_t ` *out_filter; :ref:`tapi_job_channel_t ` *out_filter2; :ref:`CHECK_RC `(:ref:`tapi_job_factory_rpc_create `(pco, &factory)); :ref:`CHECK_RC `(:ref:`tapi_job_create `(factory, "pty_spawner", "/usr/bin/tool", (const char *[]){"tool", "arg1", "arg2", NULL}, NULL, &job)); :ref:`CHECK_RC `(:ref:`tapi_job_alloc_output_channels `(job, 1, &out_channel)); :ref:`CHECK_RC `(:ref:`tapi_job_attach_filter `(:ref:`TAPI_JOB_CHANNEL_SET `(out_channel), "Readable filter", true, 0, &out_filter)); :ref:`CHECK_RC `(:ref:`tapi_job_attach_filter `(:ref:`TAPI_JOB_CHANNEL_SET `(out_channel), "Readable filter2", true, 0, &out_filter2)); :ref:`CHECK_RC `(:ref:`tapi_job_start `(job)); :ref:`CHECK_RC `(:ref:`tapi_job_wait `(job, BIG_TIMEOUT, NULL)); // Get data from the first run from both filters eos_count = 0; while (eos_count < 2) { :ref:`te_string_reset `(&buf.data); :ref:`CHECK_RC `(:ref:`tapi_job_receive `(:ref:`TAPI_JOB_CHANNEL_SET `(out_filter, out_filter2), -1, &buf)); if (buf.eos) eos_count++; ... } :ref:`CHECK_RC `(:ref:`tapi_job_start `(job)); :ref:`CHECK_RC `(:ref:`tapi_job_wait `(job, BIG_TIMEOUT, NULL)); // Get data from the second run from both filters eos_count = 0; while (eos_count < 2) { :ref:`te_string_reset `(&buf.data); :ref:`CHECK_RC `(:ref:`tapi_job_receive `(:ref:`TAPI_JOB_CHANNEL_SET `(out_filter, out_filter2), -1, &buf)); if (buf.eos) eos_count++; ... } :ref:`CHECK_RC `(:ref:`tapi_job_destroy `(job, -1)); :ref:`tapi_job_factory_destroy `(factory);