ClientConnection.cpp 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. /*
  2. * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "ClientConnection.h"
  7. #include <ConfigServer/ConfigClientEndpoint.h>
  8. #include <LibCore/ConfigFile.h>
  9. #include <LibCore/FileWatcher.h>
  10. #include <LibCore/Timer.h>
  11. namespace ConfigServer {
  12. static HashMap<int, RefPtr<ClientConnection>> s_connections;
  13. struct CachedDomain {
  14. String domain;
  15. NonnullRefPtr<Core::ConfigFile> config;
  16. RefPtr<Core::FileWatcher> watcher;
  17. };
  18. static HashMap<String, NonnullOwnPtr<CachedDomain>> s_cache;
  19. static constexpr int s_disk_sync_delay_ms = 5'000;
  20. static Core::ConfigFile& ensure_domain_config(String const& domain)
  21. {
  22. auto it = s_cache.find(domain);
  23. if (it != s_cache.end())
  24. return *it->value->config;
  25. auto config = Core::ConfigFile::open_for_app(domain, Core::ConfigFile::AllowWriting::Yes);
  26. // FIXME: Use a single FileWatcher with multiple watches inside.
  27. auto watcher_or_error = Core::FileWatcher::create(InodeWatcherFlags::Nonblock);
  28. VERIFY(!watcher_or_error.is_error());
  29. auto result = watcher_or_error.value()->add_watch(config->filename(), Core::FileWatcherEvent::Type::ContentModified);
  30. VERIFY(!result.is_error());
  31. watcher_or_error.value()->on_change = [config, domain](auto&) {
  32. auto new_config = Core::ConfigFile::open(config->filename(), Core::ConfigFile::AllowWriting::Yes);
  33. // FIXME: Detect removed keys.
  34. // FIXME: Detect type of keys.
  35. for (auto& group : new_config->groups()) {
  36. for (auto& key : new_config->keys(group)) {
  37. auto old_value = config->read_entry(group, key);
  38. auto new_value = new_config->read_entry(group, key);
  39. if (old_value != new_value) {
  40. for (auto& it : s_connections) {
  41. if (it.value->is_monitoring_domain(domain)) {
  42. it.value->async_notify_changed_string_value(domain, group, key, new_value);
  43. }
  44. }
  45. }
  46. }
  47. }
  48. // FIXME: Refactor this whole thing so that we don't need a cache lookup here.
  49. s_cache.get(domain).value()->config = new_config;
  50. };
  51. auto cache_entry = make<CachedDomain>(domain, config, watcher_or_error.release_value());
  52. s_cache.set(domain, move(cache_entry));
  53. return *config;
  54. }
  55. ClientConnection::ClientConnection(NonnullRefPtr<Core::LocalSocket> client_socket, int client_id)
  56. : IPC::ClientConnection<ConfigClientEndpoint, ConfigServerEndpoint>(*this, move(client_socket), client_id)
  57. , m_sync_timer(Core::Timer::create_single_shot(s_disk_sync_delay_ms, [this]() { sync_dirty_domains_to_disk(); }))
  58. {
  59. s_connections.set(client_id, *this);
  60. }
  61. ClientConnection::~ClientConnection()
  62. {
  63. }
  64. void ClientConnection::die()
  65. {
  66. s_connections.remove(client_id());
  67. m_sync_timer->stop();
  68. sync_dirty_domains_to_disk();
  69. }
  70. void ClientConnection::pledge_domains(Vector<String> const& domains)
  71. {
  72. if (m_has_pledged) {
  73. did_misbehave("Tried to pledge domains twice.");
  74. return;
  75. }
  76. m_has_pledged = true;
  77. for (auto& domain : domains)
  78. m_pledged_domains.set(domain);
  79. }
  80. void ClientConnection::monitor_domain(String const& domain)
  81. {
  82. if (m_has_pledged && !m_pledged_domains.contains(domain)) {
  83. did_misbehave("Attempt to monitor non-pledged domain");
  84. return;
  85. }
  86. m_monitored_domains.set(domain);
  87. }
  88. bool ClientConnection::validate_access(String const& domain, String const& group, String const& key)
  89. {
  90. if (!m_has_pledged)
  91. return true;
  92. if (m_pledged_domains.contains(domain))
  93. return true;
  94. did_misbehave(String::formatted("Blocked attempt to access domain '{}', group={}, key={}", domain, group, key).characters());
  95. return false;
  96. }
  97. void ClientConnection::sync_dirty_domains_to_disk()
  98. {
  99. if (m_dirty_domains.is_empty())
  100. return;
  101. auto dirty_domains = move(m_dirty_domains);
  102. dbgln("Syncing {} dirty domains to disk", dirty_domains.size());
  103. for (auto domain : dirty_domains) {
  104. auto& config = ensure_domain_config(domain);
  105. config.sync();
  106. }
  107. }
  108. Messages::ConfigServer::ReadStringValueResponse ClientConnection::read_string_value(String const& domain, String const& group, String const& key)
  109. {
  110. if (!validate_access(domain, group, key))
  111. return nullptr;
  112. auto& config = ensure_domain_config(domain);
  113. if (!config.has_key(group, key))
  114. return Optional<String> {};
  115. return Optional<String> { config.read_entry(group, key) };
  116. }
  117. Messages::ConfigServer::ReadI32ValueResponse ClientConnection::read_i32_value(String const& domain, String const& group, String const& key)
  118. {
  119. if (!validate_access(domain, group, key))
  120. return nullptr;
  121. auto& config = ensure_domain_config(domain);
  122. if (!config.has_key(group, key))
  123. return Optional<i32> {};
  124. return Optional<i32> { config.read_num_entry(group, key) };
  125. }
  126. Messages::ConfigServer::ReadBoolValueResponse ClientConnection::read_bool_value(String const& domain, String const& group, String const& key)
  127. {
  128. if (!validate_access(domain, group, key))
  129. return nullptr;
  130. auto& config = ensure_domain_config(domain);
  131. if (!config.has_key(group, key))
  132. return Optional<bool> {};
  133. return Optional<bool> { config.read_bool_entry(group, key) };
  134. }
  135. void ClientConnection::start_or_restart_sync_timer()
  136. {
  137. if (m_sync_timer->is_active())
  138. m_sync_timer->restart();
  139. else
  140. m_sync_timer->start();
  141. }
  142. void ClientConnection::write_string_value(String const& domain, String const& group, String const& key, String const& value)
  143. {
  144. if (!validate_access(domain, group, key))
  145. return;
  146. auto& config = ensure_domain_config(domain);
  147. if (config.has_key(group, key) && config.read_entry(group, key) == value)
  148. return;
  149. config.write_entry(group, key, value);
  150. m_dirty_domains.set(domain);
  151. start_or_restart_sync_timer();
  152. for (auto& it : s_connections) {
  153. if (it.value != this && it.value->m_monitored_domains.contains(domain))
  154. it.value->async_notify_changed_string_value(domain, group, key, value);
  155. }
  156. }
  157. void ClientConnection::write_i32_value(String const& domain, String const& group, String const& key, i32 value)
  158. {
  159. if (!validate_access(domain, group, key))
  160. return;
  161. auto& config = ensure_domain_config(domain);
  162. if (config.has_key(group, key) && config.read_num_entry(group, key) == value)
  163. return;
  164. config.write_num_entry(group, key, value);
  165. m_dirty_domains.set(domain);
  166. start_or_restart_sync_timer();
  167. for (auto& it : s_connections) {
  168. if (it.value != this && it.value->m_monitored_domains.contains(domain))
  169. it.value->async_notify_changed_i32_value(domain, group, key, value);
  170. }
  171. }
  172. void ClientConnection::write_bool_value(String const& domain, String const& group, String const& key, bool value)
  173. {
  174. if (!validate_access(domain, group, key))
  175. return;
  176. auto& config = ensure_domain_config(domain);
  177. if (config.has_key(group, key) && config.read_bool_entry(group, key) == value)
  178. return;
  179. config.write_bool_entry(group, key, value);
  180. m_dirty_domains.set(domain);
  181. start_or_restart_sync_timer();
  182. for (auto& it : s_connections) {
  183. if (it.value != this && it.value->m_monitored_domains.contains(domain))
  184. it.value->async_notify_changed_bool_value(domain, group, key, value);
  185. }
  186. }
  187. }