Skip to content

Commit

Permalink
Mon 155376 native windows service (#1899)
Browse files Browse the repository at this point in the history
* native_check_service

add robot test

* fix robot tests
  • Loading branch information
jean-christophe81 committed Dec 11, 2024
1 parent cf59e2b commit 8e48966
Show file tree
Hide file tree
Showing 20 changed files with 1,947 additions and 28 deletions.
45 changes: 27 additions & 18 deletions .github/scripts/agent_robot_test.ps1
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@
Write-Host "Work in" $pwd.ToString()

$current_dir = (pwd).Path
$wsl_path = "/mnt/" + $current_dir.SubString(0,1).ToLower() + "/" + $current_dir.SubString(3).replace('\','/')
$wsl_path = "/mnt/" + $current_dir.SubString(0, 1).ToLower() + "/" + $current_dir.SubString(3).replace('\', '/')

mkdir reports

Expand Down Expand Up @@ -102,29 +102,38 @@ Start-Process -FilePath build_windows\agent\Release\centagent.exe -ArgumentList

$uptime = (Get-WmiObject -Class Win32_OperatingSystem).LastBootUpTime #dtmf format
$d_uptime = [Management.ManagementDateTimeConverter]::ToDateTime($uptime) #datetime format
$ts_uptime = ([DateTimeOffset]$d_uptime).ToUnixTimeSeconds() #timestamp format
$ts_uptime = ([DateTimeOffset]$d_uptime).ToUnixTimeSeconds() #timestamp format

$systeminfo_data = systeminfo /FO CSV | ConvertFrom-Csv
$memory_info = @{
'total' = $systeminfo_data.'Total Physical Memory'
'free' = $systeminfo_data.'Available Physical Memory'
'virtual_max' = $systeminfo_data.'Virtual Memory: Max Size'
$snapshot = @{
'total' = $systeminfo_data.'Total Physical Memory'
'free' = $systeminfo_data.'Available Physical Memory'
'virtual_max' = $systeminfo_data.'Virtual Memory: Max Size'
'virtual_free' = $systeminfo_data.'Virtual Memory: Available'
}

$serv_list = Get-Service

$serv_stat = @{
'services.running.count' = ($serv_list | Where-Object { $_.Status -eq "Running" } | measure).Count
'services.stopped.count' = ($serv_list | Where-Object { $_.Status -eq "stopped" } | measure).Count
}

$test_param = @{
'host'= $my_host_name
'ip'= $my_ip
'wsl_path'= $wsl_path
'pwsh_path'= $pwsh_path
'drive' = @()
'current_dir' = $current_dir.replace('\','/')
'uptime' = $ts_uptime
'mem_info' = $memory_info}

Get-PSDrive -PSProvider FileSystem | Select Name, Used, Free | ForEach-Object -Process {$test_param.drive += $_}

$json_test_param = $test_param | ConvertTo-Json -Compress
'host' = $my_host_name
'ip' = $my_ip
'wsl_path' = $wsl_path
'pwsh_path' = $pwsh_path
'drive' = @()
'current_dir' = $current_dir.replace('\', '/')
'uptime' = $ts_uptime
'mem_info' = $snapshot
'serv_stat' = $serv_stat
}

Get-PSDrive -PSProvider FileSystem | Select Name, Used, Free | ForEach-Object -Process { $test_param.drive += $_ }

$json_test_param = $test_param | ConvertTo-Json -Compress

Write-Host "json_test_param" $json_test_param
$quoted_json_test_param = "'" + $json_test_param + "'"
Expand Down
13 changes: 12 additions & 1 deletion .github/workflows/windows-agent-robot-test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,18 @@ jobs:
package_extension: deb
runner: collect
arch: amd64
secrets: inherit
is_nightly: ${{ needs.get-environment.outputs.is_nightly }}
secrets:
collect_s3_access_key: ${{ secrets.COLLECT_S3_ACCESS_KEY }}
collect_s3_secret_key: ${{ secrets.COLLECT_S3_SECRET_KEY }}
registry_username: ${{ secrets.HARBOR_CENTREON_PULL_USERNAME }}
registry_password: ${{ secrets.HARBOR_CENTREON_PULL_TOKEN }}
rpm_gpg_key: ${{ secrets.RPM_GPG_SIGNING_KEY }}
rpm_gpg_signing_key_id: ${{ secrets.RPM_GPG_SIGNING_KEY_ID }}
rpm_gpg_signing_passphrase: ${{ secrets.RPM_GPG_SIGNING_PASSPHRASE }}
jira_base_url: ${{ secrets.JIRA_BASE_URL }}
jira_user_email: ${{ secrets.XRAY_JIRA_USER_EMAIL }}
jira_api_token: ${{ secrets.XRAY_JIRA_TOKEN }}


build-agent-and-execute-test:
Expand Down
1 change: 1 addition & 0 deletions agent/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -129,6 +129,7 @@ set( SRC_WINDOWS
${NATIVE_SRC}/check_uptime.cc
${NATIVE_SRC}/check_drive_size.cc
${NATIVE_SRC}/check_memory.cc
${NATIVE_SRC}/check_service.cc
)

set( SRC_LINUX
Expand Down
193 changes: 193 additions & 0 deletions agent/native_windows/inc/com/centreon/agent/check_service.hh
Original file line number Diff line number Diff line change
@@ -0,0 +1,193 @@
/**
* Copyright 2024 Centreon
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* For more information : [email protected]
*/

#ifndef CENTREON_AGENT_NATIVE_CHECK_SERVICE_HH
#define CENTREON_AGENT_NATIVE_CHECK_SERVICE_HH

#include "native_check_base.hh"

namespace com::centreon::agent {
namespace native_check_detail {

enum e_service_metric : unsigned {
stopped,
start_pending,
stop_pending,
running,
continue_pending,
pause_pending,
paused,
total,
nb_service_metric
};

/**
* @brief service filter
* it can filter services by their name and also by their start_auto
*/
class service_filter {
using string_set = absl::flat_hash_set<std::string>;

string_set _name_cache_allowed;
string_set _name_cache_excluded;
string_set _display_cache_allowed;
string_set _display_cache_excluded;

std::unique_ptr<re2::RE2> _name_filter, _name_filter_exclude;
std::unique_ptr<re2::RE2> _display_filter, _display_filter_exclude;

std::optional<bool> _start_auto;

public:
service_filter(const rapidjson::Value& args);

bool is_allowed(bool start_auto,
const std::string_view& service_name,
const std::string_view& service_display);

bool use_start_auto_filter() const { return _start_auto.has_value(); }
};

/**
* @brief service enumerator
* enumerate services and call a callback on each service allowed by filter
*/
class service_enumerator {
public:
using listener = std::function<void(const ENUM_SERVICE_STATUSA&)>;
using constructor = std::function<std::unique_ptr<service_enumerator>()>;

private:
template <bool start_auto>
void _enumerate_services(service_filter& filter,
listener&& callback,
const std::shared_ptr<spdlog::logger>& logger);

protected:
static constexpr size_t service_array_size = 512;

SC_HANDLE _sc_manager_handler = nullptr;
DWORD _resume_handle = 0;

using serv_array = ENUM_SERVICE_STATUSA[service_array_size];

virtual bool _enumerate_services(serv_array& services,
DWORD* services_returned);

virtual bool _query_service_config(
LPCSTR service_name,
QUERY_SERVICE_CONFIGA& serv_conf,
const std::shared_ptr<spdlog::logger>& logger);

public:
service_enumerator();

void reset_resume_handle() { _resume_handle = 0; }

virtual ~service_enumerator();

void enumerate_services(service_filter& filter,
listener&& callback,
const std::shared_ptr<spdlog::logger>& logger);
};

/**
* snapshot of services informations, used to create output and perfdatas
*/
class w_service_info : public snapshot<e_service_metric::nb_service_metric> {
std::string _output;
unsigned _state_to_warning;
unsigned _state_to_critical;
e_status _status = e_status::ok;

public:
w_service_info(service_enumerator& service_enumerator,
service_filter& filter,
unsigned state_to_warning,
unsigned state_to_critical,
const std::shared_ptr<spdlog::logger>& logger);

void on_service(const ENUM_SERVICE_STATUSA& service_status);

e_status get_status() const { return _status; }

void dump_to_output(std::string* output) const override;
};

} // namespace native_check_detail

/**
* @brief native final check object
*
*/
class check_service
: public native_check_base<
native_check_detail::e_service_metric::nb_service_metric> {
/**
* @brief these enums are indexed by service states values
* https://learn.microsoft.com/en-us/windows/win32/api/winsvc/ns-winsvc-service_status
*/
enum state_mask : unsigned {
stopped = 1,
start_pending = 2,
stop_pending = 4,
running = 8,
continue_pending = 16,
pause_pending = 32,
paused = 64
};

static const std::array<std::pair<std::string_view, state_mask>, 7>
_label_state;

unsigned _state_to_warning = 0;
unsigned _state_to_critical = 0;
native_check_detail::service_filter _filter;
std::unique_ptr<native_check_detail::service_enumerator> _enumerator;

public:
/**
* in order to mock services API, this static constructor is used to replace
* service_enumarator by a mock
*/
static native_check_detail::service_enumerator::constructor
_enumerator_constructor;

check_service(const std::shared_ptr<asio::io_context>& io_context,
const std::shared_ptr<spdlog::logger>& logger,
time_point first_start_expected,
duration check_interval,
const std::string& serv,
const std::string& cmd_name,
const std::string& cmd_line,
const rapidjson::Value& args,
const engine_to_agent_request_ptr& cnf,
check::completion_handler&& handler);

std::shared_ptr<native_check_detail::snapshot<
native_check_detail::e_service_metric::nb_service_metric>>
measure() override;

static void help(std::ostream& help_stream);

const std::vector<native_check_detail::metric_definition>&
get_metric_definitions() const override;
};

} // namespace com::centreon::agent
#endif
Loading

0 comments on commit 8e48966

Please sign in to comment.