369 lines
8.6 KiB
C++
369 lines
8.6 KiB
C++
// Copyright (c) 2014-2017, The Monero Project
|
|
//
|
|
// All rights reserved.
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without modification, are
|
|
// permitted provided that the following conditions are met:
|
|
//
|
|
// 1. Redistributions of source code must retain the above copyright notice, this list of
|
|
// conditions and the following disclaimer.
|
|
//
|
|
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
|
|
// of conditions and the following disclaimer in the documentation and/or other
|
|
// materials provided with the distribution.
|
|
//
|
|
// 3. Neither the name of the copyright holder nor the names of its contributors may be
|
|
// used to endorse or promote products derived from this software without specific
|
|
// prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
|
|
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
|
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
|
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
#undef UNICODE
|
|
#undef _UNICODE
|
|
|
|
#include "common/scoped_message_writer.h"
|
|
#include "daemonizer/windows_service.h"
|
|
#include "string_tools.h"
|
|
#include <chrono>
|
|
#include <iostream>
|
|
#include <utility>
|
|
#include <memory>
|
|
#include <shellapi.h>
|
|
#include <thread>
|
|
#include <windows.h>
|
|
|
|
namespace windows {
|
|
|
|
namespace {
|
|
typedef std::unique_ptr<std::remove_pointer<SC_HANDLE>::type, decltype(&::CloseServiceHandle)> service_handle;
|
|
|
|
std::string get_last_error()
|
|
{
|
|
LPSTR p_error_text = nullptr;
|
|
|
|
FormatMessage(
|
|
FORMAT_MESSAGE_FROM_SYSTEM
|
|
| FORMAT_MESSAGE_ALLOCATE_BUFFER
|
|
| FORMAT_MESSAGE_IGNORE_INSERTS
|
|
, nullptr
|
|
, GetLastError()
|
|
, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT)
|
|
, reinterpret_cast<LPSTR>(&p_error_text)
|
|
, 0
|
|
, nullptr
|
|
);
|
|
|
|
if (nullptr == p_error_text)
|
|
{
|
|
return "";
|
|
}
|
|
else
|
|
{
|
|
return std::string{p_error_text};
|
|
LocalFree(p_error_text);
|
|
}
|
|
}
|
|
|
|
bool relaunch_as_admin(
|
|
std::string const & command
|
|
, std::string const & arguments
|
|
)
|
|
{
|
|
SHELLEXECUTEINFO info{};
|
|
info.cbSize = sizeof(info);
|
|
info.lpVerb = "runas";
|
|
info.lpFile = command.c_str();
|
|
info.lpParameters = arguments.c_str();
|
|
info.hwnd = nullptr;
|
|
info.nShow = SW_SHOWNORMAL;
|
|
if (!ShellExecuteEx(&info))
|
|
{
|
|
tools::fail_msg_writer() << "Admin relaunch failed: " << get_last_error();
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// When we relaunch as admin, Windows opens a new window. This just pauses
|
|
// to allow the user to read any output.
|
|
void pause_to_display_admin_window_messages()
|
|
{
|
|
boost::chrono::milliseconds how_long{1500};
|
|
boost::this_thread::sleep_for(how_long);
|
|
}
|
|
}
|
|
|
|
bool check_admin(bool & result)
|
|
{
|
|
BOOL is_admin = FALSE;
|
|
PSID p_administrators_group = nullptr;
|
|
|
|
SID_IDENTIFIER_AUTHORITY nt_authority = SECURITY_NT_AUTHORITY;
|
|
|
|
if (!AllocateAndInitializeSid(
|
|
&nt_authority
|
|
, 2
|
|
, SECURITY_BUILTIN_DOMAIN_RID
|
|
, DOMAIN_ALIAS_RID_ADMINS
|
|
, 0, 0, 0, 0, 0, 0
|
|
, &p_administrators_group
|
|
))
|
|
{
|
|
tools::fail_msg_writer() << "Security Identifier creation failed: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
if (!CheckTokenMembership(
|
|
nullptr
|
|
, p_administrators_group
|
|
, &is_admin
|
|
))
|
|
{
|
|
tools::fail_msg_writer() << "Permissions check failed: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
result = is_admin ? true : false;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ensure_admin(
|
|
std::string const & arguments
|
|
)
|
|
{
|
|
bool admin;
|
|
|
|
if (!check_admin(admin))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (admin)
|
|
{
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
std::string command = epee::string_tools::get_current_module_path();
|
|
relaunch_as_admin(command, arguments);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool install_service(
|
|
std::string const & service_name
|
|
, std::string const & arguments
|
|
)
|
|
{
|
|
std::string command = epee::string_tools::get_current_module_path();
|
|
std::string full_command = command + arguments;
|
|
|
|
service_handle p_manager{
|
|
OpenSCManager(
|
|
nullptr
|
|
, nullptr
|
|
, SC_MANAGER_CONNECT | SC_MANAGER_CREATE_SERVICE
|
|
)
|
|
, &::CloseServiceHandle
|
|
};
|
|
if (p_manager == nullptr)
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't connect to service manager: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
service_handle p_service{
|
|
CreateService(
|
|
p_manager.get()
|
|
, service_name.c_str()
|
|
, service_name.c_str()
|
|
, 0
|
|
//, GENERIC_EXECUTE | GENERIC_READ
|
|
, SERVICE_WIN32_OWN_PROCESS
|
|
, SERVICE_DEMAND_START
|
|
, SERVICE_ERROR_NORMAL
|
|
, full_command.c_str()
|
|
, nullptr
|
|
, nullptr
|
|
, ""
|
|
//, "NT AUTHORITY\\LocalService"
|
|
, nullptr // Implies LocalSystem account
|
|
, nullptr
|
|
)
|
|
, &::CloseServiceHandle
|
|
};
|
|
if (p_service == nullptr)
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't create service: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
tools::success_msg_writer() << "Service installed";
|
|
|
|
pause_to_display_admin_window_messages();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool start_service(
|
|
std::string const & service_name
|
|
)
|
|
{
|
|
tools::msg_writer() << "Starting service";
|
|
|
|
SERVICE_STATUS_PROCESS service_status = {};
|
|
DWORD unused = 0;
|
|
|
|
service_handle p_manager{
|
|
OpenSCManager(
|
|
nullptr
|
|
, nullptr
|
|
, SC_MANAGER_CONNECT
|
|
)
|
|
, &::CloseServiceHandle
|
|
};
|
|
if (p_manager == nullptr)
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't connect to service manager: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
service_handle p_service{
|
|
OpenService(
|
|
p_manager.get()
|
|
, service_name.c_str()
|
|
//, SERVICE_START | SERVICE_QUERY_STATUS
|
|
, SERVICE_START
|
|
)
|
|
, &::CloseServiceHandle
|
|
};
|
|
if (p_service == nullptr)
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't find service: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
if (!StartService(
|
|
p_service.get()
|
|
, 0
|
|
, nullptr
|
|
))
|
|
{
|
|
tools::fail_msg_writer() << "Service start request failed: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
tools::success_msg_writer() << "Service started";
|
|
|
|
pause_to_display_admin_window_messages();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool stop_service(
|
|
std::string const & service_name
|
|
)
|
|
{
|
|
tools::msg_writer() << "Stopping service";
|
|
|
|
service_handle p_manager{
|
|
OpenSCManager(
|
|
nullptr
|
|
, nullptr
|
|
, SC_MANAGER_CONNECT
|
|
)
|
|
, &::CloseServiceHandle
|
|
};
|
|
if (p_manager == nullptr)
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't connect to service manager: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
service_handle p_service{
|
|
OpenService(
|
|
p_manager.get()
|
|
, service_name.c_str()
|
|
, SERVICE_STOP | SERVICE_QUERY_STATUS
|
|
)
|
|
, &::CloseServiceHandle
|
|
};
|
|
if (p_service == nullptr)
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't find service: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
SERVICE_STATUS status = {};
|
|
if (!ControlService(p_service.get(), SERVICE_CONTROL_STOP, &status))
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't request service stop: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
tools::success_msg_writer() << "Service stopped";
|
|
|
|
pause_to_display_admin_window_messages();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool uninstall_service(
|
|
std::string const & service_name
|
|
)
|
|
{
|
|
service_handle p_manager{
|
|
OpenSCManager(
|
|
nullptr
|
|
, nullptr
|
|
, SC_MANAGER_CONNECT
|
|
)
|
|
, &::CloseServiceHandle
|
|
};
|
|
if (p_manager == nullptr)
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't connect to service manager: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
service_handle p_service{
|
|
OpenService(
|
|
p_manager.get()
|
|
, service_name.c_str()
|
|
, SERVICE_QUERY_STATUS | DELETE
|
|
)
|
|
, &::CloseServiceHandle
|
|
};
|
|
if (p_service == nullptr)
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't find service: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
SERVICE_STATUS status = {};
|
|
if (!DeleteService(p_service.get()))
|
|
{
|
|
tools::fail_msg_writer() << "Couldn't uninstall service: " << get_last_error();
|
|
return false;
|
|
}
|
|
|
|
tools::success_msg_writer() << "Service uninstalled";
|
|
|
|
pause_to_display_admin_window_messages();
|
|
|
|
return true;
|
|
}
|
|
|
|
} // namespace windows
|