NetworkSettingsWidget.cpp 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. /*
  2. * Copyright (c) 2022, Maciej <sppmacd@pm.me>
  3. * Copyright (c) 2023, Fabian Dellwing <fabian@dellwing.net>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include "NetworkSettingsWidget.h"
  8. #include <AK/ByteString.h>
  9. #include <AK/IPv4Address.h>
  10. #include <AK/JsonObject.h>
  11. #include <AK/JsonParser.h>
  12. #include <LibCore/Command.h>
  13. #include <LibCore/System.h>
  14. #include <LibGUI/CheckBox.h>
  15. #include <LibGUI/ComboBox.h>
  16. #include <LibGUI/ItemListModel.h>
  17. #include <LibGUI/MessageBox.h>
  18. #include <LibGUI/Process.h>
  19. #include <LibGUI/SpinBox.h>
  20. #include <LibGUI/TextBox.h>
  21. #include <unistd.h>
  22. namespace NetworkSettings {
  23. static int netmask_to_cidr(IPv4Address const& address)
  24. {
  25. auto address_in_host_representation = AK::convert_between_host_and_network_endian(address.to_u32());
  26. return 32 - count_trailing_zeroes_safe(address_in_host_representation);
  27. }
  28. ErrorOr<void> NetworkSettingsWidget::initialize()
  29. {
  30. m_adapters_combobox = *find_descendant_of_type_named<GUI::ComboBox>("adapters_combobox");
  31. m_enabled_checkbox = *find_descendant_of_type_named<GUI::CheckBox>("enabled_checkbox");
  32. m_enabled_checkbox->on_checked = [&](bool value) {
  33. m_current_adapter_data->enabled = value;
  34. on_switch_enabled_or_dhcp();
  35. set_modified(true);
  36. };
  37. m_dhcp_checkbox = *find_descendant_of_type_named<GUI::CheckBox>("dhcp_checkbox");
  38. m_dhcp_checkbox->on_checked = [&](bool value) {
  39. m_current_adapter_data->dhcp = value;
  40. on_switch_enabled_or_dhcp();
  41. set_modified(true);
  42. };
  43. m_ip_address_textbox = *find_descendant_of_type_named<GUI::TextBox>("ip_address_textbox");
  44. m_ip_address_textbox->on_change = [&]() {
  45. m_current_adapter_data->ip_address = m_ip_address_textbox->text();
  46. set_modified(true);
  47. };
  48. m_cidr_spinbox = *find_descendant_of_type_named<GUI::SpinBox>("cidr_spinbox");
  49. m_cidr_spinbox->on_change = [&](int value) {
  50. m_current_adapter_data->cidr = value;
  51. set_modified(true);
  52. };
  53. m_default_gateway_textbox = *find_descendant_of_type_named<GUI::TextBox>("default_gateway_textbox");
  54. m_default_gateway_textbox->on_change = [&]() {
  55. m_current_adapter_data->default_gateway = m_default_gateway_textbox->text();
  56. set_modified(true);
  57. };
  58. auto config_file = TRY(Core::ConfigFile::open_for_system("Network"));
  59. auto proc_net_adapters_file = TRY(Core::File::open("/sys/kernel/net/adapters"sv, Core::File::OpenMode::Read));
  60. auto data = TRY(proc_net_adapters_file->read_until_eof());
  61. JsonParser parser(data);
  62. JsonValue proc_net_adapters_json = TRY(parser.parse());
  63. size_t selected_adapter_index = 0;
  64. size_t index = 0;
  65. proc_net_adapters_json.as_array().for_each([&](auto& value) {
  66. auto& if_object = value.as_object();
  67. auto adapter_name = if_object.get_byte_string("name"sv).value();
  68. if (adapter_name == "loop")
  69. return;
  70. bool adapter_exists_in_config = config_file->has_group(adapter_name);
  71. bool enabled = config_file->read_bool_entry(adapter_name, "Enabled", true);
  72. if (enabled)
  73. selected_adapter_index = index;
  74. NetworkAdapterData adapter_data;
  75. adapter_data.enabled = enabled;
  76. adapter_data.dhcp = config_file->read_bool_entry(adapter_name, "DHCP", !adapter_exists_in_config);
  77. adapter_data.ip_address = config_file->read_entry(adapter_name, "IPv4Address");
  78. auto netmask = IPv4Address::from_string(config_file->read_entry(adapter_name, "IPv4Netmask"));
  79. adapter_data.cidr = netmask.has_value() ? netmask_to_cidr(*netmask) : 32;
  80. adapter_data.default_gateway = config_file->read_entry(adapter_name, "IPv4Gateway");
  81. m_network_adapters.set(adapter_name, move(adapter_data));
  82. m_adapter_names.append(adapter_name);
  83. index++;
  84. });
  85. // FIXME: This should be done before creating a window.
  86. if (m_adapter_names.is_empty()) {
  87. GUI::MessageBox::show_error(window(), "No network adapters found!"sv);
  88. ::exit(1);
  89. }
  90. m_adapters_combobox->set_model(GUI::ItemListModel<ByteString>::create(m_adapter_names));
  91. m_adapters_combobox->on_change = [this](ByteString const& text, GUI::ModelIndex const&) {
  92. on_switch_adapter(text);
  93. };
  94. auto const& selected_adapter = selected_adapter_index;
  95. dbgln("{} in {}", selected_adapter, m_adapter_names);
  96. m_adapters_combobox->set_selected_index(selected_adapter);
  97. on_switch_adapter(m_adapter_names[selected_adapter_index]);
  98. return {};
  99. }
  100. void NetworkSettingsWidget::on_switch_adapter(ByteString const& adapter)
  101. {
  102. auto& adapter_data = m_network_adapters.get(adapter).value();
  103. m_current_adapter_data = &adapter_data;
  104. on_switch_enabled_or_dhcp();
  105. m_enabled_checkbox->set_checked(adapter_data.enabled, GUI::AllowCallback::No);
  106. m_dhcp_checkbox->set_checked(adapter_data.dhcp, GUI::AllowCallback::No);
  107. m_ip_address_textbox->set_text(adapter_data.ip_address, GUI::AllowCallback::No);
  108. m_cidr_spinbox->set_value(adapter_data.cidr, GUI::AllowCallback::No);
  109. m_default_gateway_textbox->set_text(adapter_data.default_gateway, GUI::AllowCallback::No);
  110. VERIFY(m_current_adapter_data);
  111. }
  112. void NetworkSettingsWidget::on_switch_enabled_or_dhcp()
  113. {
  114. m_dhcp_checkbox->set_enabled(m_current_adapter_data->enabled);
  115. m_ip_address_textbox->set_enabled(m_current_adapter_data->enabled && !m_current_adapter_data->dhcp);
  116. m_cidr_spinbox->set_enabled(m_current_adapter_data->enabled && !m_current_adapter_data->dhcp);
  117. m_default_gateway_textbox->set_enabled(m_current_adapter_data->enabled && !m_current_adapter_data->dhcp);
  118. }
  119. void NetworkSettingsWidget::apply_settings()
  120. {
  121. auto result = apply_settings_impl();
  122. if (result.is_error()) {
  123. GUI::MessageBox::show_error(window(), result.release_error().string_literal());
  124. return;
  125. }
  126. }
  127. ErrorOr<void> NetworkSettingsWidget::apply_settings_impl()
  128. {
  129. auto maybe_json = TRY(create_settings_object());
  130. if (!maybe_json.has_value() || maybe_json.value().is_empty())
  131. return {};
  132. auto json = maybe_json.release_value();
  133. auto pipefds = TRY(Core::System::pipe2(O_CLOEXEC));
  134. ScopeGuard guard_fd1 { [&] { close(pipefds[1]); } };
  135. {
  136. posix_spawn_file_actions_t file_actions;
  137. posix_spawn_file_actions_init(&file_actions);
  138. posix_spawn_file_actions_adddup2(&file_actions, pipefds[0], STDIN_FILENO);
  139. ScopeGuard guard_fd0_and_file_actions { [&]() {
  140. posix_spawn_file_actions_destroy(&file_actions);
  141. close(pipefds[0]);
  142. } };
  143. char const* argv[] = { "/bin/Escalator", "-I", "-P", "To apply these changes please enter your password:", "/bin/network-settings", nullptr };
  144. (void)TRY(Core::System::posix_spawn("/bin/Escalator"sv, &file_actions, nullptr, const_cast<char**>(argv), environ));
  145. auto outfile = TRY(Core::File::adopt_fd(pipefds[1], Core::File::OpenMode::Write, Core::File::ShouldCloseFileDescriptor::No));
  146. TRY(outfile->write_until_depleted(json.serialized<StringBuilder>()));
  147. }
  148. return {};
  149. }
  150. ErrorOr<Optional<JsonObject>> NetworkSettingsWidget::create_settings_object()
  151. {
  152. auto json = JsonObject();
  153. for (auto const& adapter_data : m_network_adapters) {
  154. auto netmask = TRY(IPv4Address::netmask_from_cidr(adapter_data.value.cidr).to_string());
  155. if (adapter_data.value.enabled && !adapter_data.value.dhcp) {
  156. if (!IPv4Address::from_string(adapter_data.value.ip_address).has_value()) {
  157. GUI::MessageBox::show_error(window(), TRY(String::formatted("Invalid IPv4 address for adapter {}", adapter_data.key)));
  158. return Optional<JsonObject> {};
  159. }
  160. if (!IPv4Address::from_string(adapter_data.value.default_gateway).has_value()) {
  161. GUI::MessageBox::show_error(window(), TRY(String::formatted("Invalid IPv4 gateway for adapter {}", adapter_data.key)));
  162. return Optional<JsonObject> {};
  163. }
  164. }
  165. auto adapter = JsonObject();
  166. adapter.set("Enabled", adapter_data.value.enabled);
  167. adapter.set("DHCP", adapter_data.value.dhcp);
  168. adapter.set("IPv4Address", adapter_data.value.ip_address);
  169. adapter.set("IPv4Netmask", netmask.to_byte_string());
  170. adapter.set("IPv4Gateway", adapter_data.value.default_gateway);
  171. json.set(adapter_data.key, move(adapter));
  172. }
  173. return json;
  174. }
  175. void NetworkSettingsWidget::switch_adapter(ByteString const& adapter)
  176. {
  177. m_adapters_combobox->set_text(adapter);
  178. }
  179. }