DeviceDetailsWindow.xaml.cs 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059
  1. using System.Globalization;
  2. using System.Collections.ObjectModel;
  3. using System.ComponentModel;
  4. using System.Runtime.CompilerServices;
  5. using System.Windows;
  6. using System.Windows.Controls;
  7. using System.Windows.Media;
  8. using System.Windows.Media.Animation;
  9. using NetworkTool.Client.Models;
  10. using NetworkTool.Client.Services;
  11. namespace NetworkTool.Client;
  12. public partial class DeviceDetailsWindow : Window
  13. {
  14. private const int ApplyConfirmationTimeoutSeconds = 20;
  15. private readonly ServerApiService _serverApiService = new();
  16. private readonly ObservableCollection<EditableAddress> _addresses = [];
  17. private readonly ObservableCollection<EditableRoute> _routes = [];
  18. private readonly ObservableCollection<EditableDns> _dns = [];
  19. private readonly string _baseAddress;
  20. private readonly string _localIPv4;
  21. private readonly string _password;
  22. private bool _configValidated;
  23. private bool _configDirty;
  24. private bool _isBusy;
  25. private bool _isRestoringInterfaceSelection;
  26. private bool _suppressConfigChangeHandling;
  27. private RemoteInterfaceInfo? _currentSelectedInterface;
  28. private CancellationTokenSource? _statusMessageCts;
  29. public DeviceDetailsWindow(string baseAddress, string localIPv4, string password)
  30. {
  31. InitializeComponent();
  32. AddressesDataGrid.ItemsSource = _addresses;
  33. RoutesDataGrid.ItemsSource = _routes;
  34. DnsDataGrid.ItemsSource = _dns;
  35. _baseAddress = baseAddress;
  36. _localIPv4 = localIPv4;
  37. _password = password;
  38. Loaded += DeviceDetailsWindow_OnLoaded;
  39. }
  40. private async void DeviceDetailsWindow_OnLoaded(object sender, RoutedEventArgs e)
  41. {
  42. try
  43. {
  44. await LoadRemoteDetailsAsync();
  45. UpdateButtonStates();
  46. }
  47. catch (Exception ex)
  48. {
  49. ShowStatusMessage($"读取设备信息失败:{ex.Message}");
  50. SetBusyState(false);
  51. }
  52. }
  53. private async Task LoadRemoteDetailsAsync()
  54. {
  55. ClearDetails();
  56. var device = await _serverApiService.GetDeviceInfoAsync(_baseAddress, _password, _localIPv4);
  57. if (device is not null)
  58. {
  59. RemoteDeviceIdTextBlock.Text = device.DeviceId;
  60. RemoteHostnameTextBlock.Text = device.Hostname;
  61. RemoteOsVersionTextBlock.Text = device.OSVersion;
  62. RemoteServerVersionTextBlock.Text = device.ServerVersion;
  63. }
  64. var interfaces = await _serverApiService.GetInterfacesAsync(_baseAddress, _password, _localIPv4);
  65. if (interfaces is null)
  66. {
  67. ShowStatusMessage("设备已连接,但暂时无法读取 Linux 接口列表。");
  68. return;
  69. }
  70. ShowStatusMessage($"当前管理接口:{interfaces.ManagementInterface}。请选择需要配置的目标接口。");
  71. var suggested = interfaces.Interfaces.FirstOrDefault(item => item.SystemName == interfaces.SuggestedTargetInterface)
  72. ?? interfaces.Interfaces.FirstOrDefault(item => item.IsSuggestedTarget)
  73. ?? interfaces.Interfaces.FirstOrDefault(item => !item.IsManagementInterface);
  74. RemoteTargetInterfaceTabControl.ItemsSource = interfaces.Interfaces;
  75. if (suggested is not null)
  76. {
  77. RemoteTargetInterfaceTabControl.SelectedItem = suggested;
  78. await LoadRemoteInterfaceConfigAsync(suggested.SystemName);
  79. _currentSelectedInterface = suggested;
  80. }
  81. }
  82. private void ClearDetails()
  83. {
  84. RemoteDeviceIdTextBlock.Text = "-";
  85. RemoteHostnameTextBlock.Text = "-";
  86. RemoteOsVersionTextBlock.Text = "-";
  87. RemoteServerVersionTextBlock.Text = "-";
  88. RemoteTargetInterfaceTabControl.ItemsSource = null;
  89. _addresses.Clear();
  90. _routes.Clear();
  91. _dns.Clear();
  92. DefaultGatewayCheckBox.IsChecked = false;
  93. DefaultGatewayTextBox.Text = string.Empty;
  94. CustomRoutesCheckBox.IsChecked = false;
  95. _configValidated = false;
  96. _configDirty = false;
  97. _currentSelectedInterface = null;
  98. }
  99. private async void RemoteTargetInterfaceTabControl_OnSelectionChanged(object sender, SelectionChangedEventArgs e)
  100. {
  101. try
  102. {
  103. if (RemoteTargetInterfaceTabControl.SelectedItem is not RemoteInterfaceInfo selected)
  104. {
  105. UpdateButtonStates();
  106. return;
  107. }
  108. if (_isRestoringInterfaceSelection)
  109. {
  110. return;
  111. }
  112. if (_configDirty && _currentSelectedInterface is not null && selected.SystemName != _currentSelectedInterface.SystemName)
  113. {
  114. var result = MessageBox.Show(
  115. this,
  116. "当前配置已修改,切换接口会丢失未应用内容。是否继续?",
  117. "确认切换接口",
  118. MessageBoxButton.OKCancel,
  119. MessageBoxImage.Warning);
  120. if (result != MessageBoxResult.OK)
  121. {
  122. _isRestoringInterfaceSelection = true;
  123. RemoteTargetInterfaceTabControl.SelectedItem = _currentSelectedInterface;
  124. _isRestoringInterfaceSelection = false;
  125. return;
  126. }
  127. }
  128. await LoadRemoteInterfaceConfigAsync(selected.SystemName, useBusyState: true);
  129. _currentSelectedInterface = selected;
  130. }
  131. catch (Exception ex)
  132. {
  133. ShowStatusMessage($"读取目标接口配置失败:{ex.Message}");
  134. SetBusyState(false);
  135. }
  136. }
  137. private async Task LoadRemoteInterfaceConfigAsync(string interfaceName, bool useBusyState = false)
  138. {
  139. if (useBusyState)
  140. {
  141. SetBusyState(true, "正在读取 Linux 端 IP 配置...");
  142. }
  143. try
  144. {
  145. var result = await _serverApiService.GetInterfaceConfigAsync(_baseAddress, _password, _localIPv4, interfaceName);
  146. if (!result.Success || result.Data is null)
  147. {
  148. ShowStatusMessage($"读取目标接口 {interfaceName} 配置失败:{result.Message}");
  149. return;
  150. }
  151. var config = result.Data;
  152. _suppressConfigChangeHandling = true;
  153. Dhcp4CheckBox.IsChecked = false;
  154. _addresses.Clear();
  155. foreach (var address in config.EffectiveAddresses)
  156. {
  157. _addresses.Add(new EditableAddress { IP = address.IP, Mask = PrefixToMask(address.Prefix) });
  158. }
  159. _routes.Clear();
  160. DefaultGatewayCheckBox.IsChecked = false;
  161. DefaultGatewayTextBox.Text = string.Empty;
  162. foreach (var route in config.EffectiveRoutes)
  163. {
  164. if (route.To.Equals("default", StringComparison.OrdinalIgnoreCase))
  165. {
  166. DefaultGatewayCheckBox.IsChecked = true;
  167. DefaultGatewayTextBox.Text = route.Via;
  168. }
  169. else
  170. {
  171. _routes.Add(new EditableRoute { To = route.To, Via = route.Via });
  172. }
  173. }
  174. CustomRoutesCheckBox.IsChecked = _routes.Count > 0;
  175. _dns.Clear();
  176. if (config.Dns is not null)
  177. {
  178. foreach (var dns in config.Dns)
  179. {
  180. _dns.Add(new EditableDns { Address = dns });
  181. }
  182. }
  183. _suppressConfigChangeHandling = false;
  184. _configValidated = false;
  185. _configDirty = false;
  186. ShowStatusMessage("已读取Linux端IP配置。");
  187. UpdateButtonStates();
  188. }
  189. finally
  190. {
  191. if (useBusyState)
  192. {
  193. SetBusyState(false);
  194. }
  195. }
  196. }
  197. private async void ReloadInterfaceConfigButton_OnClick(object sender, RoutedEventArgs e)
  198. {
  199. if (RemoteTargetInterfaceTabControl.SelectedItem is RemoteInterfaceInfo selected)
  200. {
  201. await LoadRemoteInterfaceConfigAsync(selected.SystemName);
  202. }
  203. }
  204. private async void ValidateConfigButton_OnClick(object sender, RoutedEventArgs e)
  205. {
  206. if (RemoteTargetInterfaceTabControl.SelectedItem is not RemoteInterfaceInfo selected)
  207. {
  208. return;
  209. }
  210. var request = BuildConfigRequest(selected.SystemName);
  211. if (request is null)
  212. {
  213. return;
  214. }
  215. SetBusyState(true, "正在校验配置,请稍候...");
  216. try
  217. {
  218. var result = await _serverApiService.ValidateInterfaceConfigAsync(_baseAddress, _password, _localIPv4, request);
  219. _configValidated = result.Success && result.Data?.Valid == true;
  220. if (result.Data is not null)
  221. {
  222. var warnings = result.Data.Warnings.Count > 0 ? $" 警告:{string.Join(";", result.Data.Warnings)}" : string.Empty;
  223. var errors = result.Data.Errors.Count > 0 ? $" 错误:{string.Join(";", result.Data.Errors)}" : string.Empty;
  224. ShowStatusMessage(_configValidated ? $"校验通过,可应用配置。{warnings}" : $"校验失败。{errors}{warnings}");
  225. }
  226. else
  227. {
  228. ShowStatusMessage($"校验失败:{result.Message}");
  229. }
  230. UpdateButtonStates();
  231. }
  232. finally
  233. {
  234. SetBusyState(false);
  235. }
  236. }
  237. private async void ApplyConfigButton_OnClick(object sender, RoutedEventArgs e)
  238. {
  239. if (RemoteTargetInterfaceTabControl.SelectedItem is not RemoteInterfaceInfo selected)
  240. {
  241. return;
  242. }
  243. var request = BuildConfigRequest(selected.SystemName);
  244. if (request is null)
  245. {
  246. return;
  247. }
  248. var confirmMessage = $"将要把以下配置应用到接口 {selected.SystemName}:\n\n" +
  249. $"模式:{(request.Dhcp4 ? "DHCP 自动获取" : "静态 IPv4")}\n" +
  250. $"IP:{(request.Dhcp4 ? "自动获取" : FormatAddresses(request.Addresses))}\n" +
  251. $"路由:{(request.Dhcp4 ? "自动获取" : FormatRoutes(request.Routes))}\n" +
  252. $"DNS:{(request.Dns.Count == 0 ? "无" : string.Join(", ", request.Dns))}\n\n" +
  253. "请确认是否继续。";
  254. if (MessageBox.Show(this, confirmMessage, "确认应用配置", MessageBoxButton.OKCancel, MessageBoxImage.Question) != MessageBoxResult.OK)
  255. {
  256. return;
  257. }
  258. SetBusyState(true, "正在提交并应用配置,请稍候...");
  259. try
  260. {
  261. var applyResult = await _serverApiService.ApplyInterfaceConfigAsync(_baseAddress, _password, _localIPv4, request);
  262. if (!applyResult.Success || applyResult.Data is null)
  263. {
  264. ShowStatusMessage($"提交配置任务失败:{applyResult.Message}");
  265. return;
  266. }
  267. ShowStatusMessage("配置任务已提交,正在应用并等待连通确认...");
  268. await PollTaskAsync(applyResult.Data.TaskId);
  269. }
  270. finally
  271. {
  272. SetBusyState(false);
  273. }
  274. }
  275. private async Task PollTaskAsync(string taskId)
  276. {
  277. var transientFailureCount = 0;
  278. var confirmationRequested = false;
  279. for (var i = 0; i < 20; i++)
  280. {
  281. await Task.Delay(1000);
  282. var result = await _serverApiService.GetTaskAsync(_baseAddress, _password, _localIPv4, taskId);
  283. if (!result.Success || result.Data is null)
  284. {
  285. if (result.StatusCode is null)
  286. {
  287. transientFailureCount++;
  288. ShowStatusMessage($"设备连接短暂中断,正在重试({transientFailureCount})。");
  289. continue;
  290. }
  291. ShowStatusMessage($"读取任务状态失败:{result.Message}");
  292. return;
  293. }
  294. transientFailureCount = 0;
  295. var task = result.Data;
  296. ShowStatusMessage(FormatTaskStatusMessage(task));
  297. if (task.Status == "running" && task.Step == "confirming" && !confirmationRequested)
  298. {
  299. confirmationRequested = true;
  300. var confirm = ShowApplyConfirmationDialog(ApplyConfirmationTimeoutSeconds);
  301. if (confirm)
  302. {
  303. var confirmResult = await _serverApiService.ConfirmApplyTaskAsync(_baseAddress, _password, _localIPv4, taskId);
  304. ShowStatusMessage(confirmResult.Success ? "已发送保留配置确认。" : $"发送确认失败:{confirmResult.Message}");
  305. }
  306. else
  307. {
  308. var cancelResult = await _serverApiService.CancelApplyTaskAsync(_baseAddress, _password, _localIPv4, taskId);
  309. ShowStatusMessage(cancelResult.Success ? "已取消保留配置,正在回滚。" : $"发送取消失败:{cancelResult.Message}");
  310. }
  311. }
  312. if (task.Status is "success" or "failed" or "rolled_back")
  313. {
  314. ShowTaskCompletionDialog(task);
  315. if (RemoteTargetInterfaceTabControl.SelectedItem is RemoteInterfaceInfo selected)
  316. {
  317. await LoadRemoteInterfaceConfigAsync(selected.SystemName);
  318. }
  319. return;
  320. }
  321. }
  322. ShowStatusMessage($"任务 {taskId} 轮询超时,请稍后手动刷新。");
  323. }
  324. private bool ShowApplyConfirmationDialog(int timeoutSeconds)
  325. {
  326. var remaining = timeoutSeconds;
  327. var result = false;
  328. var messageTextBlock = new TextBlock
  329. {
  330. Width = 420,
  331. TextWrapping = TextWrapping.Wrap,
  332. FontSize = 13,
  333. Foreground = Brushes.Black,
  334. };
  335. var confirmButton = new Button
  336. {
  337. MinWidth = 88,
  338. MinHeight = 32,
  339. Margin = new Thickness(0, 0, 10, 0),
  340. Content = "确认保留",
  341. IsDefault = true,
  342. };
  343. var cancelButton = new Button
  344. {
  345. MinWidth = 88,
  346. MinHeight = 32,
  347. Content = "取消回滚",
  348. IsCancel = true,
  349. };
  350. var dialog = new Window
  351. {
  352. Title = "确认保留网络配置",
  353. Owner = this,
  354. WindowStartupLocation = WindowStartupLocation.CenterOwner,
  355. ResizeMode = ResizeMode.NoResize,
  356. SizeToContent = SizeToContent.WidthAndHeight,
  357. Content = new StackPanel
  358. {
  359. Margin = new Thickness(18),
  360. Children =
  361. {
  362. messageTextBlock,
  363. new StackPanel
  364. {
  365. Margin = new Thickness(0, 18, 0, 0),
  366. HorizontalAlignment = HorizontalAlignment.Right,
  367. Orientation = Orientation.Horizontal,
  368. Children = { confirmButton, cancelButton },
  369. },
  370. },
  371. },
  372. };
  373. void UpdateMessage()
  374. {
  375. messageTextBlock.Text = $"当前客户端仍可连接到设备。是否确认保留这次网络配置?\n\n剩余 {remaining} 秒;超时或取消时,Linux 端会自动回滚。";
  376. }
  377. var timer = new System.Windows.Threading.DispatcherTimer { Interval = TimeSpan.FromSeconds(1) };
  378. timer.Tick += (_, _) =>
  379. {
  380. remaining--;
  381. if (remaining <= 0)
  382. {
  383. timer.Stop();
  384. dialog.DialogResult = false;
  385. dialog.Close();
  386. return;
  387. }
  388. UpdateMessage();
  389. };
  390. confirmButton.Click += (_, _) =>
  391. {
  392. result = true;
  393. dialog.DialogResult = true;
  394. dialog.Close();
  395. };
  396. cancelButton.Click += (_, _) =>
  397. {
  398. dialog.DialogResult = false;
  399. dialog.Close();
  400. };
  401. dialog.Closed += (_, _) => timer.Stop();
  402. UpdateMessage();
  403. timer.Start();
  404. dialog.ShowDialog();
  405. return result;
  406. }
  407. private async void RebootButton_OnClick(object sender, RoutedEventArgs e)
  408. {
  409. await ExecuteSystemActionAsync(
  410. "重启设备",
  411. "设备将立即重启,当前窗口和连接可能马上中断。是否继续?",
  412. () => _serverApiService.RebootAsync(_baseAddress, _password, _localIPv4));
  413. }
  414. private async void ShutdownButton_OnClick(object sender, RoutedEventArgs e)
  415. {
  416. await ExecuteSystemActionAsync(
  417. "关闭设备",
  418. "设备将立即关机,当前窗口和连接可能马上中断。是否继续?",
  419. () => _serverApiService.ShutdownAsync(_baseAddress, _password, _localIPv4));
  420. }
  421. private async Task ExecuteSystemActionAsync(string title, string confirmMessage, Func<Task<ApiCallResult<RemoteSystemActionResponse>>> action)
  422. {
  423. if (MessageBox.Show(this, confirmMessage, title, MessageBoxButton.OKCancel, MessageBoxImage.Warning) != MessageBoxResult.OK)
  424. {
  425. return;
  426. }
  427. var result = await action();
  428. if (!result.Success || result.Data is null)
  429. {
  430. ShowStatusMessage($"{title}失败:{result.Message}");
  431. return;
  432. }
  433. ShowStatusMessage($"{title}任务已提交:{result.Data.TaskId}。命令已发出,设备可能立即断开。");
  434. }
  435. private RemoteInterfaceConfig? BuildConfigRequest(string interfaceName)
  436. {
  437. CommitConfigEdits();
  438. var dhcp4 = Dhcp4CheckBox.IsChecked == true;
  439. var addresses = Array.Empty<RemoteInterfaceAddressConfig>();
  440. var routes = Array.Empty<RemoteInterfaceRouteConfig>();
  441. if (!dhcp4)
  442. {
  443. if (_addresses.All(item => string.IsNullOrWhiteSpace(item.IP) && string.IsNullOrWhiteSpace(item.Mask)))
  444. {
  445. ShowStatusMessage("IP 地址不能为空,至少需要填写一行地址。");
  446. return null;
  447. }
  448. if (!TryBuildAddresses(out addresses, out var addressError))
  449. {
  450. ShowStatusMessage(addressError);
  451. return null;
  452. }
  453. if (!TryBuildRoutes(out routes, out var routeError))
  454. {
  455. ShowStatusMessage(routeError);
  456. return null;
  457. }
  458. }
  459. var dns = _dns.Select(item => item.Address.Trim()).Where(item => item != string.Empty).ToArray();
  460. return new RemoteInterfaceConfig
  461. {
  462. Interface = interfaceName,
  463. Dhcp4 = dhcp4,
  464. Addresses = dhcp4 ? Array.Empty<RemoteInterfaceAddressConfig>() : addresses,
  465. Routes = dhcp4 ? Array.Empty<RemoteInterfaceRouteConfig>() : routes,
  466. Dns = dhcp4 ? Array.Empty<string>() : dns,
  467. };
  468. }
  469. private void CommitConfigEdits()
  470. {
  471. AddressesDataGrid.CommitEdit(DataGridEditingUnit.Cell, true);
  472. AddressesDataGrid.CommitEdit(DataGridEditingUnit.Row, true);
  473. RoutesDataGrid.CommitEdit(DataGridEditingUnit.Cell, true);
  474. RoutesDataGrid.CommitEdit(DataGridEditingUnit.Row, true);
  475. DnsDataGrid.CommitEdit(DataGridEditingUnit.Cell, true);
  476. DnsDataGrid.CommitEdit(DataGridEditingUnit.Row, true);
  477. }
  478. private static string FormatCurrentIp(RemoteInterfaceConfig config)
  479. {
  480. if (config.EffectiveAddresses.Count == 0)
  481. {
  482. return config.Dhcp4 ? "DHCP 自动获取,暂无 IPv4" : "无";
  483. }
  484. var text = FormatAddresses(config.EffectiveAddresses);
  485. return config.Dhcp4 ? $"{text} (DHCP)" : text;
  486. }
  487. private static string FormatAddresses(IReadOnlyList<RemoteInterfaceAddressConfig> addresses)
  488. {
  489. return addresses.Count == 0 ? "无" : string.Join(Environment.NewLine, addresses.Select(item => $"{item.IP}/{item.Prefix}"));
  490. }
  491. private static string FormatRoutes(IReadOnlyList<RemoteInterfaceRouteConfig> routes)
  492. {
  493. return routes.Count == 0 ? "无" : string.Join(Environment.NewLine, routes.Select(item => $"{item.To} via {item.Via}"));
  494. }
  495. private bool TryBuildAddresses(out RemoteInterfaceAddressConfig[] addresses, out string error)
  496. {
  497. var result = new List<RemoteInterfaceAddressConfig>();
  498. foreach (var row in _addresses)
  499. {
  500. var ip = row.IP.Trim();
  501. var maskText = row.Mask.Trim();
  502. if (ip == string.Empty && maskText == string.Empty)
  503. {
  504. continue;
  505. }
  506. if (ip.Contains('/'))
  507. {
  508. var parts = ip.Split('/', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries);
  509. if (parts.Length != 2 || !int.TryParse(parts[1], out var cidrPrefix) || cidrPrefix < 0 || cidrPrefix > 32)
  510. {
  511. addresses = [];
  512. error = $"地址格式不正确:{ip}";
  513. return false;
  514. }
  515. row.IP = parts[0];
  516. row.Mask = PrefixToMask(cidrPrefix);
  517. result.Add(new RemoteInterfaceAddressConfig { IP = parts[0], Prefix = cidrPrefix });
  518. continue;
  519. }
  520. if (ip == string.Empty || maskText == string.Empty)
  521. {
  522. addresses = [];
  523. error = "IP 地址和子网掩码都需要填写。";
  524. return false;
  525. }
  526. if (!TryMaskOrPrefixToPrefix(maskText, out var prefix))
  527. {
  528. addresses = [];
  529. error = $"子网掩码格式不正确:{ip} {maskText}";
  530. return false;
  531. }
  532. row.Mask = PrefixToMask(prefix);
  533. result.Add(new RemoteInterfaceAddressConfig { IP = ip, Prefix = prefix });
  534. }
  535. addresses = result.ToArray();
  536. error = string.Empty;
  537. return addresses.Length > 0;
  538. }
  539. private bool TryBuildRoutes(out RemoteInterfaceRouteConfig[] routes, out string error)
  540. {
  541. var result = new List<RemoteInterfaceRouteConfig>();
  542. if (DefaultGatewayCheckBox.IsChecked == true)
  543. {
  544. var gateway = DefaultGatewayTextBox.Text.Trim();
  545. if (gateway == string.Empty)
  546. {
  547. routes = [];
  548. error = "启用默认网关时,网关地址不能为空。";
  549. return false;
  550. }
  551. result.Add(new RemoteInterfaceRouteConfig { To = "default", Via = gateway });
  552. }
  553. if (CustomRoutesCheckBox.IsChecked == true)
  554. {
  555. foreach (var row in _routes)
  556. {
  557. var to = row.To.Trim();
  558. var via = row.Via.Trim();
  559. if (to == string.Empty && via == string.Empty)
  560. {
  561. continue;
  562. }
  563. if (to == string.Empty || via == string.Empty)
  564. {
  565. routes = [];
  566. error = "自定义路由的目标网段和网关地址都需要填写。";
  567. return false;
  568. }
  569. result.Add(new RemoteInterfaceRouteConfig { To = to, Via = via });
  570. }
  571. }
  572. routes = result.ToArray();
  573. error = string.Empty;
  574. return true;
  575. }
  576. private static bool TryParseAddresses(string text, out RemoteInterfaceAddressConfig[] addresses, out string error)
  577. {
  578. var result = new List<RemoteInterfaceAddressConfig>();
  579. foreach (var line in ParseListText(text))
  580. {
  581. var parts = line.Split(' ', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries);
  582. if (parts.Length == 1 && line.Contains('/'))
  583. {
  584. var cidrParts = line.Split('/', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries);
  585. if (cidrParts.Length != 2 || !int.TryParse(cidrParts[1], out var prefix) || prefix < 0 || prefix > 32)
  586. {
  587. addresses = [];
  588. error = $"地址格式不正确:{line}";
  589. return false;
  590. }
  591. result.Add(new RemoteInterfaceAddressConfig { IP = cidrParts[0], Prefix = prefix });
  592. continue;
  593. }
  594. if (parts.Length != 2)
  595. {
  596. addresses = [];
  597. error = $"地址格式不正确:{line}";
  598. return false;
  599. }
  600. if (!TryMaskOrPrefixToPrefix(parts[1], out var parsedPrefix))
  601. {
  602. addresses = [];
  603. error = $"子网掩码或前缀格式不正确:{line}";
  604. return false;
  605. }
  606. result.Add(new RemoteInterfaceAddressConfig { IP = parts[0], Prefix = parsedPrefix });
  607. }
  608. addresses = result.ToArray();
  609. error = string.Empty;
  610. return addresses.Length > 0;
  611. }
  612. private static bool TryParseRoutes(string text, out RemoteInterfaceRouteConfig[] routes, out string error)
  613. {
  614. var result = new List<RemoteInterfaceRouteConfig>();
  615. foreach (var line in ParseListText(text))
  616. {
  617. var parts = line.Split(' ', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries);
  618. if (parts.Length == 1)
  619. {
  620. result.Add(new RemoteInterfaceRouteConfig { To = "default", Via = parts[0] });
  621. continue;
  622. }
  623. if (parts.Length == 3 && parts[1].Equals("via", StringComparison.OrdinalIgnoreCase))
  624. {
  625. result.Add(new RemoteInterfaceRouteConfig { To = parts[0], Via = parts[2] });
  626. continue;
  627. }
  628. routes = [];
  629. error = $"路由格式不正确:{line}";
  630. return false;
  631. }
  632. routes = result.ToArray();
  633. error = string.Empty;
  634. return true;
  635. }
  636. private static string[] ParseListText(string text)
  637. {
  638. return text.Split(['\r', '\n', ',', ';'], StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries);
  639. }
  640. private static bool TryMaskOrPrefixToPrefix(string text, out int prefix)
  641. {
  642. if (int.TryParse(text, out prefix) && prefix >= 0 && prefix <= 32)
  643. {
  644. return true;
  645. }
  646. return TryMaskToPrefix(text, out prefix);
  647. }
  648. private static string PrefixToMask(int prefix)
  649. {
  650. if (prefix < 0 || prefix > 32)
  651. {
  652. return string.Empty;
  653. }
  654. var mask = prefix == 0 ? 0u : uint.MaxValue << (32 - prefix);
  655. return string.Join('.', new[] { (mask >> 24) & 255, (mask >> 16) & 255, (mask >> 8) & 255, mask & 255 });
  656. }
  657. private static bool TryMaskToPrefix(string maskText, out int prefix)
  658. {
  659. prefix = 0;
  660. if (string.IsNullOrWhiteSpace(maskText))
  661. {
  662. return false;
  663. }
  664. var parts = maskText.Trim().Split('.');
  665. if (parts.Length != 4)
  666. {
  667. return false;
  668. }
  669. uint mask = 0;
  670. foreach (var part in parts)
  671. {
  672. if (!byte.TryParse(part, out var octet))
  673. {
  674. return false;
  675. }
  676. mask = (mask << 8) | octet;
  677. }
  678. var seenZero = false;
  679. for (var i = 31; i >= 0; i--)
  680. {
  681. var bit = (mask & (1u << i)) != 0;
  682. if (bit && seenZero)
  683. {
  684. return false;
  685. }
  686. if (bit)
  687. {
  688. prefix++;
  689. }
  690. else
  691. {
  692. seenZero = true;
  693. }
  694. }
  695. return true;
  696. }
  697. private void ConfigInputChanged_OnChanged(object sender, TextChangedEventArgs e)
  698. {
  699. if (_suppressConfigChangeHandling)
  700. {
  701. return;
  702. }
  703. _configValidated = false;
  704. ShowStatusMessage("配置内容已变更,请重新点击“2. 校验配置”。");
  705. UpdateButtonStates();
  706. }
  707. private void ConfigModeChanged_OnChanged(object sender, RoutedEventArgs e)
  708. {
  709. if (_suppressConfigChangeHandling)
  710. {
  711. UpdateButtonStates();
  712. return;
  713. }
  714. _configValidated = false;
  715. _configDirty = true;
  716. ShowStatusMessage("配置模式已变更,请重新点击“2. 校验配置”。");
  717. UpdateButtonStates();
  718. }
  719. private void GatewayOrRouteModeChanged_OnChanged(object sender, RoutedEventArgs e)
  720. {
  721. if (_suppressConfigChangeHandling)
  722. {
  723. UpdateButtonStates();
  724. return;
  725. }
  726. _configValidated = false;
  727. _configDirty = true;
  728. ShowStatusMessage("配置内容已变更,请重新点击“2. 校验配置”。");
  729. UpdateButtonStates();
  730. }
  731. private void ConfigGrid_OnCellEditEnding(object sender, DataGridCellEditEndingEventArgs e)
  732. {
  733. if (e.Row.Item is EditableAddress address)
  734. {
  735. NormalizeAddressRow(address);
  736. }
  737. MarkConfigChanged("配置内容已变更,请重新点击“2. 校验配置”。");
  738. }
  739. private void AddAddressButton_OnClick(object sender, RoutedEventArgs e)
  740. {
  741. _addresses.Add(new EditableAddress { Mask = "255.255.255.0" });
  742. MarkConfigChanged("已添加 IP 地址,请填写后重新校验配置。");
  743. }
  744. private void AddRouteButton_OnClick(object sender, RoutedEventArgs e)
  745. {
  746. _routes.Add(new EditableRoute());
  747. MarkConfigChanged("已添加路由,请填写后重新校验配置。");
  748. }
  749. private void AddDnsButton_OnClick(object sender, RoutedEventArgs e)
  750. {
  751. _dns.Add(new EditableDns());
  752. MarkConfigChanged("已添加 DNS,请填写后重新校验配置。");
  753. }
  754. private void DeleteAddressButton_OnClick(object sender, RoutedEventArgs e)
  755. {
  756. if ((sender as FrameworkElement)?.DataContext is not EditableAddress address)
  757. {
  758. return;
  759. }
  760. _addresses.Remove(address);
  761. MarkConfigChanged("已删除 IP 地址,请重新校验配置。");
  762. }
  763. private void DeleteRouteButton_OnClick(object sender, RoutedEventArgs e)
  764. {
  765. if ((sender as FrameworkElement)?.DataContext is EditableRoute route)
  766. {
  767. _routes.Remove(route);
  768. MarkConfigChanged("已删除路由,请重新校验配置。");
  769. }
  770. }
  771. private void DeleteDnsButton_OnClick(object sender, RoutedEventArgs e)
  772. {
  773. if ((sender as FrameworkElement)?.DataContext is EditableDns dns)
  774. {
  775. _dns.Remove(dns);
  776. MarkConfigChanged("已删除 DNS,请重新校验配置。");
  777. }
  778. }
  779. private void MarkConfigChanged(string message)
  780. {
  781. if (_suppressConfigChangeHandling)
  782. {
  783. return;
  784. }
  785. _configValidated = false;
  786. _configDirty = true;
  787. ShowStatusMessage(message);
  788. UpdateButtonStates();
  789. }
  790. private static void NormalizeAddressRow(EditableAddress row)
  791. {
  792. var ip = row.IP.Trim();
  793. if (!ip.Contains('/'))
  794. {
  795. return;
  796. }
  797. var parts = ip.Split('/', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries);
  798. if (parts.Length == 2 && int.TryParse(parts[1], out var prefix) && prefix >= 0 && prefix <= 32)
  799. {
  800. row.IP = parts[0];
  801. row.Mask = PrefixToMask(prefix);
  802. }
  803. }
  804. private void ShowStatusMessage(string message)
  805. {
  806. ApplyStatusMessageStyle(message);
  807. StatusMessageTextBlock.Text = message;
  808. StatusMessageBorder.Opacity = 0;
  809. StatusMessageBorder.Visibility = Visibility.Visible;
  810. StatusMessageBorder.BeginAnimation(OpacityProperty, new DoubleAnimation(1, TimeSpan.FromMilliseconds(160)));
  811. _statusMessageCts?.Cancel();
  812. _statusMessageCts = new CancellationTokenSource();
  813. _ = HideStatusMessageAsync(_statusMessageCts.Token);
  814. }
  815. private async Task HideStatusMessageAsync(CancellationToken cancellationToken)
  816. {
  817. try
  818. {
  819. await Task.Delay(3000, cancellationToken);
  820. await Dispatcher.InvokeAsync(() =>
  821. {
  822. var animation = new DoubleAnimation(0, TimeSpan.FromMilliseconds(200));
  823. animation.Completed += (_, _) =>
  824. {
  825. if (!cancellationToken.IsCancellationRequested)
  826. {
  827. StatusMessageBorder.Visibility = Visibility.Collapsed;
  828. }
  829. };
  830. StatusMessageBorder.BeginAnimation(OpacityProperty, animation);
  831. });
  832. }
  833. catch (TaskCanceledException)
  834. {
  835. }
  836. }
  837. private void ApplyStatusMessageStyle(string message)
  838. {
  839. var (background, foreground) = GetStatusMessageBrushes(message);
  840. StatusMessageBorder.Background = background;
  841. StatusMessageTextBlock.Foreground = foreground;
  842. }
  843. private static (Brush Background, Brush Foreground) GetStatusMessageBrushes(string message)
  844. {
  845. if (ContainsAny(message, "失败", "错误", "拒绝", "超时", "不能为空", "不正确", "无法"))
  846. {
  847. return (new SolidColorBrush((Color)ColorConverter.ConvertFromString("#B91C1C")), Brushes.White);
  848. }
  849. if (ContainsAny(message, "未发现", "请", "重试", "警告", "需要"))
  850. {
  851. return (new SolidColorBrush((Color)ColorConverter.ConvertFromString("#C2410C")), Brushes.White);
  852. }
  853. if (ContainsAny(message, "成功", "已切换", "已刷新", "已读取", "已加载", "已发现", "已提交", "已回填"))
  854. {
  855. return (new SolidColorBrush((Color)ColorConverter.ConvertFromString("#047857")), Brushes.White);
  856. }
  857. return (new SolidColorBrush((Color)ColorConverter.ConvertFromString("#111827")), Brushes.White);
  858. }
  859. private static bool ContainsAny(string message, params string[] markers)
  860. {
  861. return markers.Any(marker => message.Contains(marker, StringComparison.Ordinal));
  862. }
  863. private static string FormatTaskStatusMessage(RemoteTaskResult task)
  864. {
  865. return task.Status switch
  866. {
  867. "success" => string.IsNullOrWhiteSpace(task.Detail) ? "配置已成功应用。" : task.Detail,
  868. "failed" => string.IsNullOrWhiteSpace(task.Detail) ? "配置应用失败。" : task.Detail,
  869. "rolled_back" => string.IsNullOrWhiteSpace(task.Detail) ? "配置应用失败,已自动回滚。" : task.Detail,
  870. _ => task.Step switch
  871. {
  872. "validating" => "正在校验配置...",
  873. "writing_netplan" => "正在写入 Linux 网络配置...",
  874. "applying" => "正在应用 Linux 网络配置...",
  875. "confirming" => string.IsNullOrWhiteSpace(task.Detail) ? "等待确认保留配置..." : task.Detail,
  876. "rolling_back" => "配置应用失败,正在自动回滚...",
  877. _ => string.IsNullOrWhiteSpace(task.Detail) ? "正在处理,请稍候..." : task.Detail,
  878. }
  879. };
  880. }
  881. private void ShowTaskCompletionDialog(RemoteTaskResult task)
  882. {
  883. var message = FormatTaskStatusMessage(task);
  884. var title = task.Status == "success" ? "应用配置成功" : "应用配置失败";
  885. var image = task.Status == "success" ? MessageBoxImage.Information : MessageBoxImage.Warning;
  886. MessageBox.Show(this, message, title, MessageBoxButton.OK, image);
  887. }
  888. private void UpdateButtonStates()
  889. {
  890. var hasSelectedInterface = RemoteTargetInterfaceTabControl.SelectedItem is RemoteInterfaceInfo;
  891. var canEdit = !_isBusy && hasSelectedInterface;
  892. var canEditStatic = canEdit && Dhcp4CheckBox.IsChecked != true;
  893. var canEditGateway = canEditStatic && DefaultGatewayCheckBox.IsChecked == true;
  894. var canEditCustomRoutes = canEditStatic && CustomRoutesCheckBox.IsChecked == true;
  895. RemoteTargetInterfaceTabControl.IsEnabled = !_isBusy && RemoteTargetInterfaceTabControl.Items.Count > 0;
  896. ReloadInterfaceConfigButton.IsEnabled = canEdit;
  897. ValidateConfigButton.IsEnabled = canEdit;
  898. ApplyConfigButton.IsEnabled = !_isBusy && _configValidated && hasSelectedInterface;
  899. Dhcp4CheckBox.IsEnabled = canEdit;
  900. AddressesDataGrid.IsEnabled = canEditStatic;
  901. DefaultGatewayCheckBox.IsEnabled = canEditStatic;
  902. DefaultGatewayTextBox.IsEnabled = canEditGateway;
  903. CustomRoutesCheckBox.IsEnabled = canEditStatic;
  904. RoutesDataGrid.IsEnabled = canEditCustomRoutes;
  905. DnsDataGrid.IsEnabled = canEditStatic;
  906. AddAddressButton.IsEnabled = canEditStatic;
  907. AddRouteButton.IsEnabled = canEditCustomRoutes;
  908. AddDnsButton.IsEnabled = canEditStatic;
  909. RebootButton.IsEnabled = !_isBusy;
  910. ShutdownButton.IsEnabled = !_isBusy;
  911. }
  912. private void SetBusyState(bool isBusy, string? message = null)
  913. {
  914. _isBusy = isBusy;
  915. BusyOverlay.Visibility = isBusy ? Visibility.Visible : Visibility.Collapsed;
  916. BusyMessageTextBlock.Text = string.IsNullOrWhiteSpace(message) ? "正在处理,请稍候..." : message;
  917. UpdateButtonStates();
  918. }
  919. private sealed class EditableAddress : INotifyPropertyChanged
  920. {
  921. private string _ip = string.Empty;
  922. private string _mask = string.Empty;
  923. public string IP
  924. {
  925. get => _ip;
  926. set => SetField(ref _ip, value);
  927. }
  928. public string Mask
  929. {
  930. get => _mask;
  931. set => SetField(ref _mask, value);
  932. }
  933. public event PropertyChangedEventHandler? PropertyChanged;
  934. private void SetField<T>(ref T field, T value, [CallerMemberName] string propertyName = "")
  935. {
  936. if (EqualityComparer<T>.Default.Equals(field, value))
  937. {
  938. return;
  939. }
  940. field = value;
  941. PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
  942. }
  943. }
  944. private sealed class EditableRoute
  945. {
  946. public string To { get; set; } = string.Empty;
  947. public string Via { get; set; } = string.Empty;
  948. }
  949. private sealed class EditableDns
  950. {
  951. public string Address { get; set; } = string.Empty;
  952. }
  953. }