/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */ #include "remote/apiuser.hpp" #include "remote/apiuser-ti.cpp" #include "base/configtype.hpp" #include "base/base64.hpp" #include "base/tlsutility.hpp" #include "base/utility.hpp" #include #ifndef _WIN32 # include #endif /* _WIN32 */ using namespace icinga; REGISTER_TYPE(ApiUser); ApiUser::Ptr ApiUser::GetByClientCN(const String& cn) { for (const ApiUser::Ptr& user : ConfigType::GetObjectsByType()) { if (user->GetClientCN() == cn) return user; } return nullptr; } ApiUser::Ptr ApiUser::GetByAuthHeader(const String& auth_header) { String::SizeType pos = auth_header.FindFirstOf(" "); String username, password; if (pos != String::NPos && auth_header.SubStr(0, pos) == "Basic") { String credentials_base64 = auth_header.SubStr(pos + 1); String credentials = Base64::Decode(credentials_base64); String::SizeType cpos = credentials.FindFirstOf(":"); if (cpos != String::NPos) { username = credentials.SubStr(0, cpos); password = credentials.SubStr(cpos + 1); } } const ApiUser::Ptr& user = ApiUser::GetByName(username); /* Deny authentication if: * 1) user does not exist * 2) given password is empty * 2) configured password does not match. */ if (!user) { return nullptr; } #ifndef _WIN32 auto hash (user->GetHashedPassword()); if (!hash.IsEmpty()) { static std::mutex mtx; std::unique_lock lock (mtx); auto hashed (crypt(password.CStr(), hash.CStr())); return hashed == hash ? user : nullptr; } #endif /* _WIN32 */ auto plain (user->GetPassword()); return !plain.IsEmpty() && Utility::ComparePasswords(password, plain) ? user : nullptr; }