diff options
Diffstat (limited to 'cloud_print')
-rw-r--r-- | cloud_print/service/win/cloud_print_service.cc | 84 |
1 files changed, 72 insertions, 12 deletions
diff --git a/cloud_print/service/win/cloud_print_service.cc b/cloud_print/service/win/cloud_print_service.cc index 08ef93d..c0948c4 100644 --- a/cloud_print/service/win/cloud_print_service.cc +++ b/cloud_print/service/win/cloud_print_service.cc @@ -4,6 +4,7 @@ #include "cloud_print/service/win/cloud_print_service.h" +#include <atlsecurity.h> #include <iomanip> #include <iostream> @@ -23,6 +24,8 @@ namespace { const wchar_t kServiceStateFileName[] = L"Service State"; +const wchar_t kUserToRunService[] = L"NT AUTHORITY\\LocalService"; + // The traits class for Windows Service. class ServiceHandleTraits { public: @@ -124,6 +127,43 @@ std::string GetOption(const std::string& name, const std::string& default, return tmp; } +HRESULT WriteFileAsUser(const FilePath& path, const wchar_t* user, + const char* data, int size) { + ATL::CAccessToken thread_token; + if (!thread_token.OpenThreadToken(TOKEN_DUPLICATE | TOKEN_IMPERSONATE)) { + LOG(ERROR) << "Failed to open thread token."; + return HResultFromLastError(); + } + + ATL::CSid local_service; + if (!local_service.LoadAccount(user)) { + LOG(ERROR) << "Failed create SID."; + return HResultFromLastError(); + } + + ATL::CAccessToken token; + ATL::CTokenGroups group; + group.Add(local_service, 0); + + const ATL::CTokenGroups empty_group; + if (!thread_token.CreateRestrictedToken(&token, empty_group, group)) { + LOG(ERROR) << "Failed to create restricted token for " << user << "."; + return HResultFromLastError(); + } + + if (!token.Impersonate()) { + LOG(ERROR) << "Failed to impersonate " << user << "."; + return HResultFromLastError(); + } + + ATL::CAutoRevertImpersonation auto_revert(&token); + if (file_util::WriteFile(path, data, size) != size) { + LOG(ERROR) << "Failed to write file " << path.value() << "."; + return HResultFromLastError(); + } + return S_OK; +} + } // namespace class CloudPrintServiceModule @@ -141,7 +181,7 @@ class CloudPrintServiceModule return S_OK; } - HRESULT InstallService(const FilePath& user_data_dir) { + HRESULT InstallService() { // TODO(vitalybuka): consider "lite" version if we don't want unregister // printers here. HRESULT hr = UninstallService(); @@ -161,7 +201,7 @@ class CloudPrintServiceModule CHECK(PathService::Get(base::FILE_EXE, &service_path)); CommandLine command_line(service_path); command_line.AppendSwitch(kServiceSwitch); - command_line.AppendSwitchPath(kUserDataDirSwitch, user_data_dir); + command_line.AppendSwitchPath(kUserDataDirSwitch, user_data_dir_); ServiceHandle scm; hr = OpenServiceManager(&scm); @@ -173,7 +213,7 @@ class CloudPrintServiceModule scm, m_szServiceName, m_szServiceName, SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS, SERVICE_AUTO_START, SERVICE_ERROR_NORMAL, command_line.GetCommandLineString().c_str(), NULL, NULL, NULL, - L"NT AUTHORITY\\LocalService", NULL)); + kUserToRunService, NULL)); if (!service.IsValid()) return HResultFromLastError(); @@ -239,12 +279,15 @@ class CloudPrintServiceModule return S_FALSE; } - HRESULT hr = ProcessServiceState(user_data_dir_, - command_line.HasSwitch(kQuietSwitch)); + HRESULT hr = ValidateUserDataDir(); if (FAILED(hr)) return hr; - hr = InstallService(user_data_dir_); + hr = ProcessServiceState(command_line.HasSwitch(kQuietSwitch)); + if (FAILED(hr)) + return hr; + + hr = InstallService(); if (SUCCEEDED(hr) && command_line.HasSwitch(kStartSwitch)) return StartService(); @@ -270,8 +313,24 @@ class CloudPrintServiceModule return S_FALSE; } - HRESULT ProcessServiceState(const FilePath& user_data_dir, bool quiet) { - FilePath file = user_data_dir.Append(kServiceStateFileName); + HRESULT ValidateUserDataDir() { + FilePath temp_file; + const char some_data[] = "1234"; + if (!file_util::CreateTemporaryFileInDir(user_data_dir_, &temp_file)) + return E_FAIL; + HRESULT hr = WriteFileAsUser(temp_file, kUserToRunService, some_data, + sizeof(some_data)); + if (FAILED(hr)) { + LOG(ERROR) << "Failed to write user data. Make sure that account \'" << + kUserToRunService << "\'has full access to \'" << + user_data_dir_.value() << "\'."; + } + file_util::Delete(temp_file, false); + return hr; + } + + HRESULT ProcessServiceState(bool quiet) { + FilePath file = user_data_dir_.Append(kServiceStateFileName); for (;;) { std::string contents; @@ -315,10 +374,11 @@ class CloudPrintServiceModule if (is_valid) { std::string new_contents = service_state.ToString(); if (new_contents != contents) { - if (file_util::WriteFile(file, new_contents.c_str(), - new_contents.size()) <= 0) { - return HResultFromLastError(); - } + HRESULT hr = WriteFileAsUser(file, kUserToRunService, + new_contents.c_str(), + new_contents.size()); + if (FAILED(hr)) + return hr; } } } |