mirror of
				https://github.com/2dust/v2rayN.git
				synced 2025-10-26 18:24:43 +00:00 
			
		
		
		
	Compare commits
	
		
			12 commits
		
	
	
		
			ce438d8f18
			...
			8ac9499d8c
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 8ac9499d8c | ||
|   | 11924b545f | ||
|   | ab1dc45ed4 | ||
|   | e96a4818c4 | ||
|   | 0377e7ce19 | ||
|   | 2d41272659 | ||
|   | 6929886b3e | ||
|   | 721d70c8c7 | ||
|   | 27b45aee83 | ||
|   | e7f75010d3 | ||
|   | aa1ccdd01b | ||
|   | b17323c982 | 
					 32 changed files with 886 additions and 395 deletions
				
			
		|  | @ -20,7 +20,7 @@ | ||||||
|     <PackageVersion Include="ReactiveUI.WPF" Version="20.4.1" /> |     <PackageVersion Include="ReactiveUI.WPF" Version="20.4.1" /> | ||||||
|     <PackageVersion Include="Semi.Avalonia" Version="11.2.1.10" /> |     <PackageVersion Include="Semi.Avalonia" Version="11.2.1.10" /> | ||||||
|     <PackageVersion Include="Semi.Avalonia.DataGrid" Version="11.2.1.10" /> |     <PackageVersion Include="Semi.Avalonia.DataGrid" Version="11.2.1.10" /> | ||||||
|     <PackageVersion Include="Splat.NLog" Version="16.2.1" /> |     <PackageVersion Include="Splat.NLog" Version="17.0.1" /> | ||||||
|     <PackageVersion Include="sqlite-net-pcl" Version="1.9.172" /> |     <PackageVersion Include="sqlite-net-pcl" Version="1.9.172" /> | ||||||
|     <PackageVersion Include="TaskScheduler" Version="2.12.2" /> |     <PackageVersion Include="TaskScheduler" Version="2.12.2" /> | ||||||
|     <PackageVersion Include="WebDav.Client" Version="2.9.0" /> |     <PackageVersion Include="WebDav.Client" Version="2.9.0" /> | ||||||
|  |  | ||||||
|  | @ -6,16 +6,22 @@ namespace ServiceLib.Handler; | ||||||
| public static class AppEvents | public static class AppEvents | ||||||
| { | { | ||||||
|     public static readonly Subject<Unit> ProfilesRefreshRequested = new(); |     public static readonly Subject<Unit> ProfilesRefreshRequested = new(); | ||||||
|  |     public static readonly Subject<Unit> SubscriptionsRefreshRequested = new(); | ||||||
|  |     public static readonly Subject<Unit> ProxiesReloadRequested = new(); | ||||||
|  |     public static readonly Subject<ServerSpeedItem> DispatcherStatisticsRequested = new(); | ||||||
| 
 | 
 | ||||||
|     public static readonly Subject<string> SendSnackMsgRequested = new(); |     public static readonly Subject<string> SendSnackMsgRequested = new(); | ||||||
| 
 |  | ||||||
|     public static readonly Subject<string> SendMsgViewRequested = new(); |     public static readonly Subject<string> SendMsgViewRequested = new(); | ||||||
| 
 | 
 | ||||||
|     public static readonly Subject<Unit> AppExitRequested = new(); |     public static readonly Subject<Unit> AppExitRequested = new(); | ||||||
| 
 |  | ||||||
|     public static readonly Subject<bool> ShutdownRequested = new(); |     public static readonly Subject<bool> ShutdownRequested = new(); | ||||||
| 
 | 
 | ||||||
|     public static readonly Subject<Unit> AdjustMainLvColWidthRequested = new(); |     public static readonly Subject<Unit> AdjustMainLvColWidthRequested = new(); | ||||||
| 
 | 
 | ||||||
|     public static readonly Subject<ServerSpeedItem> DispatcherStatisticsRequested = new(); |     public static readonly Subject<string> SetDefaultServerRequested = new(); | ||||||
|  | 
 | ||||||
|  |     public static readonly Subject<Unit> RoutingsMenuRefreshRequested = new(); | ||||||
|  |     public static readonly Subject<Unit> TestServerRequested = new(); | ||||||
|  |     public static readonly Subject<Unit> InboundDisplayRequested = new(); | ||||||
|  |     public static readonly Subject<ESysProxyType> SysProxyChangeRequested = new(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1251,7 +1251,7 @@ public static class ConfigHandler | ||||||
|     public static async Task<ProfileItem?> GetPreSocksItem(Config config, ProfileItem node, ECoreType coreType) |     public static async Task<ProfileItem?> GetPreSocksItem(Config config, ProfileItem node, ECoreType coreType) | ||||||
|     { |     { | ||||||
|         ProfileItem? itemSocks = null; |         ProfileItem? itemSocks = null; | ||||||
|         if (node.ConfigType != EConfigType.Custom && node.ConfigType < EConfigType.Group && coreType != ECoreType.sing_box && config.TunModeItem.EnableTun) |         if (node.ConfigType != EConfigType.Custom && coreType != ECoreType.sing_box && config.TunModeItem.EnableTun) | ||||||
|         { |         { | ||||||
|             itemSocks = new ProfileItem() |             itemSocks = new ProfileItem() | ||||||
|             { |             { | ||||||
|  | @ -1262,7 +1262,7 @@ public static class ConfigHandler | ||||||
|                 Port = AppManager.Instance.GetLocalPort(EInboundProtocol.socks) |                 Port = AppManager.Instance.GetLocalPort(EInboundProtocol.socks) | ||||||
|             }; |             }; | ||||||
|         } |         } | ||||||
|         else if (node.ConfigType == EConfigType.Custom && node.ConfigType < EConfigType.Group && node.PreSocksPort > 0) |         else if (node.ConfigType == EConfigType.Custom && node.PreSocksPort > 0) | ||||||
|         { |         { | ||||||
|             var preCoreType = config.RunningCoreType = config.TunModeItem.EnableTun ? ECoreType.sing_box : ECoreType.Xray; |             var preCoreType = config.RunningCoreType = config.TunModeItem.EnableTun ? ECoreType.sing_box : ECoreType.Xray; | ||||||
|             itemSocks = new ProfileItem() |             itemSocks = new ProfileItem() | ||||||
|  |  | ||||||
|  | @ -32,6 +32,12 @@ public class ProfileItem : ReactiveObject | ||||||
|     public string GetSummary() |     public string GetSummary() | ||||||
|     { |     { | ||||||
|         var summary = $"[{(ConfigType).ToString()}] "; |         var summary = $"[{(ConfigType).ToString()}] "; | ||||||
|  |         if (IsComplex()) | ||||||
|  |         { | ||||||
|  |             summary += $"[{CoreType.ToString()}]{Remarks}"; | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|             var arrAddr = Address.Contains(':') ? Address.Split(':') : Address.Split('.'); |             var arrAddr = Address.Contains(':') ? Address.Split(':') : Address.Split('.'); | ||||||
|             var addr = arrAddr.Length switch |             var addr = arrAddr.Length switch | ||||||
|             { |             { | ||||||
|  | @ -39,12 +45,6 @@ public class ProfileItem : ReactiveObject | ||||||
|                 > 1 => $"***{arrAddr.Last()}", |                 > 1 => $"***{arrAddr.Last()}", | ||||||
|                 _ => Address |                 _ => Address | ||||||
|             }; |             }; | ||||||
|         if (ConfigType is EConfigType.Custom or > EConfigType.Group) |  | ||||||
|         { |  | ||||||
|             summary += $"[{CoreType.ToString()}]{Remarks}"; |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             summary += $"{Remarks}({addr}:{Port})"; |             summary += $"{Remarks}({addr}:{Port})"; | ||||||
|         } |         } | ||||||
|         return summary; |         return summary; | ||||||
|  | @ -64,6 +64,51 @@ public class ProfileItem : ReactiveObject | ||||||
|         return Network.TrimEx(); |         return Network.TrimEx(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public bool IsComplex() | ||||||
|  |     { | ||||||
|  |         return ConfigType is EConfigType.Custom or > EConfigType.Group; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public bool IsValid() | ||||||
|  |     { | ||||||
|  |         if (IsComplex()) | ||||||
|  |             return true; | ||||||
|  | 
 | ||||||
|  |         if (Address.IsNullOrEmpty() || Port is <= 0 or >= 65536) | ||||||
|  |             return false; | ||||||
|  | 
 | ||||||
|  |         switch (ConfigType) | ||||||
|  |         { | ||||||
|  |             case EConfigType.VMess: | ||||||
|  |                 if (Id.IsNullOrEmpty() || !Utils.IsGuidByParse(Id)) | ||||||
|  |                     return false; | ||||||
|  |                 break; | ||||||
|  | 
 | ||||||
|  |             case EConfigType.VLESS: | ||||||
|  |                 if (Id.IsNullOrEmpty() || (!Utils.IsGuidByParse(Id) && Id.Length > 30)) | ||||||
|  |                     return false; | ||||||
|  |                 if (!Global.Flows.Contains(Flow)) | ||||||
|  |                     return false; | ||||||
|  |                 break; | ||||||
|  | 
 | ||||||
|  |             case EConfigType.Shadowsocks: | ||||||
|  |                 if (Id.IsNullOrEmpty()) | ||||||
|  |                     return false; | ||||||
|  |                 if (string.IsNullOrEmpty(Security) || !Global.SsSecuritiesInSingbox.Contains(Security)) | ||||||
|  |                     return false; | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if ((ConfigType is EConfigType.VLESS or EConfigType.Trojan) | ||||||
|  |             && StreamSecurity == Global.StreamSecurityReality | ||||||
|  |             && PublicKey.IsNullOrEmpty()) | ||||||
|  |         { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     #endregion function |     #endregion function | ||||||
| 
 | 
 | ||||||
|     [PrimaryKey] |     [PrimaryKey] | ||||||
|  |  | ||||||
							
								
								
									
										90
									
								
								v2rayN/ServiceLib/Resx/ResUI.Designer.cs
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										90
									
								
								v2rayN/ServiceLib/Resx/ResUI.Designer.cs
									
									
									
										generated
									
									
									
								
							|  | @ -114,6 +114,33 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Core '{0}' does not support network type '{1}'. 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string CoreNotSupportNetwork { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("CoreNotSupportNetwork", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Core '{0}' does not support protocol '{1}'. 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string CoreNotSupportProtocol { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("CoreNotSupportProtocol", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Core '{0}' does not support protocol '{1}' when using transport '{2}'. 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string CoreNotSupportProtocolTransport { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("CoreNotSupportProtocolTransport", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Note that custom configuration relies entirely on your own configuration and does not work with all settings. If you want to use the system proxy, please modify the listening port manually. 的本地化字符串。 |         ///   查找类似 Note that custom configuration relies entirely on your own configuration and does not work with all settings. If you want to use the system proxy, please modify the listening port manually. 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -267,6 +294,24 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Group '{0}' is empty. Please add at least one node. 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string GroupEmpty { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("GroupEmpty", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 The group "{0}" cannot reference itself. 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string GroupSelfReference { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("GroupSelfReference", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 This is not the correct configuration, please check 的本地化字符串。 |         ///   查找类似 This is not the correct configuration, please check 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -294,6 +339,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 The {0} property is invalid, please check. 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string InvalidProperty { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("InvalidProperty", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Invalid address (URL) 的本地化字符串。 |         ///   查找类似 Invalid address (URL) 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -2004,6 +2058,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Node alias '{0}' does not exist. 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string NodeTagNotExist { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("NodeTagNotExist", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Non-VMess or SS protocol 的本地化字符串。 |         ///   查找类似 Non-VMess or SS protocol 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -2112,6 +2175,24 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Policy group:  的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string PolicyGroupPrefix { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("PolicyGroupPrefix", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Proxy chained:  的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string ProxyChainedPrefix { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("ProxyChainedPrefix", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Global hotkey {0} registration failed, reason: {1} 的本地化字符串。 |         ///   查找类似 Global hotkey {0} registration failed, reason: {1} 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -2175,6 +2256,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Routing rule outbound:  的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string RoutingRuleOutboundPrefix { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("RoutingRuleOutboundPrefix", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Run as Admin 的本地化字符串。 |         ///   查找类似 Run as Admin 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  |  | ||||||
|  | @ -1563,4 +1563,34 @@ | ||||||
|   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration Fallback by Xray</value> |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="CoreNotSupportNetwork" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support network type '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocolTransport" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}' when using transport '{2}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocol" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="ProxyChainedPrefix" xml:space="preserve"> | ||||||
|  |     <value>Proxy chained: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="RoutingRuleOutboundPrefix" xml:space="preserve"> | ||||||
|  |     <value>Routing rule outbound: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="PolicyGroupPrefix" xml:space="preserve"> | ||||||
|  |     <value>Policy group: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="NodeTagNotExist" xml:space="preserve"> | ||||||
|  |     <value>Node alias '{0}' does not exist.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupEmpty" xml:space="preserve"> | ||||||
|  |     <value>Group '{0}' is empty. Please add at least one node.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="InvalidProperty" xml:space="preserve"> | ||||||
|  |     <value>The {0} property is invalid, please check.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupSelfReference" xml:space="preserve"> | ||||||
|  |     <value>The group "{0}" cannot reference itself.</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1563,4 +1563,34 @@ | ||||||
|   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration Fallback by Xray</value> |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="CoreNotSupportNetwork" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support network type '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocolTransport" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}' when using transport '{2}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocol" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="ProxyChainedPrefix" xml:space="preserve"> | ||||||
|  |     <value>Proxy chained: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="RoutingRuleOutboundPrefix" xml:space="preserve"> | ||||||
|  |     <value>Routing rule outbound: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="PolicyGroupPrefix" xml:space="preserve"> | ||||||
|  |     <value>Policy group: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="NodeTagNotExist" xml:space="preserve"> | ||||||
|  |     <value>Node alias '{0}' does not exist.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupEmpty" xml:space="preserve"> | ||||||
|  |     <value>Group '{0}' is empty. Please add at least one node.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="InvalidProperty" xml:space="preserve"> | ||||||
|  |     <value>The {0} property is invalid, please check.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupSelfReference" xml:space="preserve"> | ||||||
|  |     <value>The group "{0}" cannot reference itself.</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1563,4 +1563,34 @@ | ||||||
|   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration Fallback by Xray</value> |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="CoreNotSupportNetwork" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support network type '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocolTransport" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}' when using transport '{2}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocol" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="ProxyChainedPrefix" xml:space="preserve"> | ||||||
|  |     <value>Proxy chained: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="RoutingRuleOutboundPrefix" xml:space="preserve"> | ||||||
|  |     <value>Routing rule outbound: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="PolicyGroupPrefix" xml:space="preserve"> | ||||||
|  |     <value>Policy group: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="NodeTagNotExist" xml:space="preserve"> | ||||||
|  |     <value>Node alias '{0}' does not exist.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupEmpty" xml:space="preserve"> | ||||||
|  |     <value>Group '{0}' is empty. Please add at least one node.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="InvalidProperty" xml:space="preserve"> | ||||||
|  |     <value>The {0} property is invalid, please check.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupSelfReference" xml:space="preserve"> | ||||||
|  |     <value>The group "{0}" cannot reference itself.</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1563,4 +1563,34 @@ | ||||||
|   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration Fallback by Xray</value> |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="CoreNotSupportNetwork" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support network type '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocolTransport" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}' when using transport '{2}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocol" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="ProxyChainedPrefix" xml:space="preserve"> | ||||||
|  |     <value>Proxy chained: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="RoutingRuleOutboundPrefix" xml:space="preserve"> | ||||||
|  |     <value>Routing rule outbound: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="PolicyGroupPrefix" xml:space="preserve"> | ||||||
|  |     <value>Policy group: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="NodeTagNotExist" xml:space="preserve"> | ||||||
|  |     <value>Node alias '{0}' does not exist.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupEmpty" xml:space="preserve"> | ||||||
|  |     <value>Group '{0}' is empty. Please add at least one node.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="InvalidProperty" xml:space="preserve"> | ||||||
|  |     <value>The {0} property is invalid, please check.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupSelfReference" xml:space="preserve"> | ||||||
|  |     <value>The group "{0}" cannot reference itself.</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1560,4 +1560,34 @@ | ||||||
|   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|     <value>多配置文件故障转移 Xray</value> |     <value>多配置文件故障转移 Xray</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="CoreNotSupportNetwork" xml:space="preserve"> | ||||||
|  |     <value>核心 '{0}' 不支持网络类型 '{1}'。</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocolTransport" xml:space="preserve"> | ||||||
|  |     <value>核心 '{0}' 在使用传输方式 '{2}' 时不支持协议 '{1}'。</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocol" xml:space="preserve"> | ||||||
|  |     <value>核心 '{0}' 不支持协议 '{1}'。</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="ProxyChainedPrefix" xml:space="preserve"> | ||||||
|  |     <value>代理链: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="RoutingRuleOutboundPrefix" xml:space="preserve"> | ||||||
|  |     <value>路由规则出站: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="PolicyGroupPrefix" xml:space="preserve"> | ||||||
|  |     <value>策略组: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="NodeTagNotExist" xml:space="preserve"> | ||||||
|  |     <value>节点别名 '{0}' 不存在。</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupEmpty" xml:space="preserve"> | ||||||
|  |     <value>组“{0}”为空。请至少添加一个节点。</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="InvalidProperty" xml:space="preserve"> | ||||||
|  |     <value>{0}属性无效,请检查</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupSelfReference" xml:space="preserve"> | ||||||
|  |     <value>{0} 分组不能引用自身</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1560,4 +1560,34 @@ | ||||||
|   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration Fallback by Xray</value> |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="CoreNotSupportNetwork" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support network type '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocolTransport" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}' when using transport '{2}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="CoreNotSupportProtocol" xml:space="preserve"> | ||||||
|  |     <value>Core '{0}' does not support protocol '{1}'.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="ProxyChainedPrefix" xml:space="preserve"> | ||||||
|  |     <value>Proxy chained: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="RoutingRuleOutboundPrefix" xml:space="preserve"> | ||||||
|  |     <value>Routing rule outbound: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="PolicyGroupPrefix" xml:space="preserve"> | ||||||
|  |     <value>Policy group: </value> | ||||||
|  |   </data> | ||||||
|  |   <data name="NodeTagNotExist" xml:space="preserve"> | ||||||
|  |     <value>Node alias '{0}' does not exist.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupEmpty" xml:space="preserve"> | ||||||
|  |     <value>Group '{0}' is empty. Please add at least one node.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="InvalidProperty" xml:space="preserve"> | ||||||
|  |     <value>The {0} property is invalid, please check.</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="GroupSelfReference" xml:space="preserve"> | ||||||
|  |     <value>The group "{0}" cannot reference itself.</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
							
								
								
									
										282
									
								
								v2rayN/ServiceLib/Services/ActionPrecheckService.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										282
									
								
								v2rayN/ServiceLib/Services/ActionPrecheckService.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,282 @@ | ||||||
|  | namespace ServiceLib.Services; | ||||||
|  | 
 | ||||||
|  | /// <summary> | ||||||
|  | /// Centralized pre-checks before sensitive actions (set active profile, generate config, etc.). | ||||||
|  | /// </summary> | ||||||
|  | public class ActionPrecheckService(Config config) | ||||||
|  | { | ||||||
|  |     private static readonly Lazy<ActionPrecheckService> _instance = new(() => new ActionPrecheckService(AppManager.Instance.Config)); | ||||||
|  |     public static ActionPrecheckService Instance => _instance.Value; | ||||||
|  | 
 | ||||||
|  |     private readonly Config _config = config; | ||||||
|  | 
 | ||||||
|  |     public async Task<List<string>> CheckBeforeSetActive(string? indexId) | ||||||
|  |     { | ||||||
|  |         if (indexId.IsNullOrEmpty()) | ||||||
|  |         { | ||||||
|  |             return [ResUI.PleaseSelectServer]; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var item = await AppManager.Instance.GetProfileItem(indexId); | ||||||
|  |         if (item is null) | ||||||
|  |         { | ||||||
|  |             return [ResUI.PleaseSelectServer]; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return await CheckBeforeGenerateConfig(item); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task<List<string>> CheckBeforeGenerateConfig(ProfileItem? item) | ||||||
|  |     { | ||||||
|  |         if (item is null) | ||||||
|  |         { | ||||||
|  |             return [ResUI.PleaseSelectServer]; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var errors = new List<string>(); | ||||||
|  | 
 | ||||||
|  |         errors.AddRange(await ValidateCurrentNodeAndCoreSupport(item)); | ||||||
|  |         errors.AddRange(await ValidateRelatedNodesExistAndValid(item)); | ||||||
|  | 
 | ||||||
|  |         return errors; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<List<string>> ValidateCurrentNodeAndCoreSupport(ProfileItem item) | ||||||
|  |     { | ||||||
|  |         if (item.ConfigType == EConfigType.Custom) | ||||||
|  |         { | ||||||
|  |             return []; | ||||||
|  |         } | ||||||
|  |         var coreType = AppManager.Instance.GetCoreType(item, item.ConfigType); | ||||||
|  |         return await ValidateNodeAndCoreSupport(item, coreType); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<List<string>> ValidateNodeAndCoreSupport(ProfileItem item, ECoreType? coreType = null) | ||||||
|  |     { | ||||||
|  |         var errors = new List<string>(); | ||||||
|  |          | ||||||
|  |         // sing-box does not support xhttp / kcp | ||||||
|  |         // sing-box does not support transports like ws/http/httpupgrade/etc. when the node is not vmess/trojan/vless | ||||||
|  |         coreType ??= AppManager.Instance.GetCoreType(item, item.ConfigType); | ||||||
|  | 
 | ||||||
|  |         if (item.ConfigType is EConfigType.Custom) | ||||||
|  |         { | ||||||
|  |             errors.Add(string.Format(ResUI.CoreNotSupportProtocol, coreType.ToString(), item.ConfigType.ToString())); | ||||||
|  |             return errors; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (!item.IsComplex()) | ||||||
|  |         { | ||||||
|  |             if (item.Address.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 errors.Add(string.Format(ResUI.InvalidProperty, "Address")); | ||||||
|  |                 return errors; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (item.Port is <= 0 or >= 65536) | ||||||
|  |             { | ||||||
|  |                 errors.Add(string.Format(ResUI.InvalidProperty, "Port")); | ||||||
|  |                 return errors; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |             switch (item.ConfigType) | ||||||
|  |             { | ||||||
|  |                 case EConfigType.VMess: | ||||||
|  |                     if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) | ||||||
|  |                         errors.Add(string.Format(ResUI.InvalidProperty, "Id")); | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 case EConfigType.VLESS: | ||||||
|  |                     if (item.Id.IsNullOrEmpty() || (!Utils.IsGuidByParse(item.Id) && item.Id.Length > 30)) | ||||||
|  |                         errors.Add(string.Format(ResUI.InvalidProperty, "Id")); | ||||||
|  |                     if (!Global.Flows.Contains(item.Flow)) | ||||||
|  |                         errors.Add(string.Format(ResUI.InvalidProperty, "Flow")); | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 case EConfigType.Shadowsocks: | ||||||
|  |                     if (item.Id.IsNullOrEmpty()) | ||||||
|  |                         errors.Add(string.Format(ResUI.InvalidProperty, "Id")); | ||||||
|  |                     if (string.IsNullOrEmpty(item.Security) || !Global.SsSecuritiesInSingbox.Contains(item.Security)) | ||||||
|  |                         errors.Add(string.Format(ResUI.InvalidProperty, "Security")); | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if ((item.ConfigType is EConfigType.VLESS or EConfigType.Trojan) | ||||||
|  |                 && item.StreamSecurity == Global.StreamSecurityReality | ||||||
|  |                 && item.PublicKey.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 errors.Add(string.Format(ResUI.InvalidProperty, "PublicKey")); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (errors.Count > 0) | ||||||
|  |             { | ||||||
|  |                 return errors; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (item.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |         { | ||||||
|  |             ProfileGroupItemManager.Instance.TryGet(item.IndexId, out var group); | ||||||
|  |             if (group is null || group.ChildItems.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 errors.Add(string.Format(ResUI.GroupEmpty, item.Remarks)); | ||||||
|  |                 return errors; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             foreach (var child in Utils.String2List(group.ChildItems)) | ||||||
|  |             { | ||||||
|  |                 var childErrors = new List<string>(); | ||||||
|  |                 if (child.IsNullOrEmpty()) | ||||||
|  |                 { | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  |                  | ||||||
|  |                 var childItem = await AppManager.Instance.GetProfileItem(child); | ||||||
|  |                 if (childItem is null) | ||||||
|  |                 { | ||||||
|  |                     childErrors.Add(string.Format(ResUI.NodeTagNotExist, child)); | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 // self-reference check | ||||||
|  |                 if (childItem.IndexId == item.IndexId) | ||||||
|  |                 { | ||||||
|  |                     childErrors.Add(string.Format(ResUI.GroupSelfReference, childItem.Remarks)); | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 if (childItem.ConfigType is EConfigType.Custom or EConfigType.ProxyChain) | ||||||
|  |                 { | ||||||
|  |                     childErrors.Add(string.Format(ResUI.InvalidProperty, childItem.Remarks)); | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 childErrors.AddRange(await ValidateNodeAndCoreSupport(childItem, coreType)); | ||||||
|  |                 errors.AddRange(childErrors); | ||||||
|  |             } | ||||||
|  |             return errors; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var net = item.GetNetwork() ?? item.Network; | ||||||
|  | 
 | ||||||
|  |         if (coreType == ECoreType.sing_box) | ||||||
|  |         { | ||||||
|  |             if (net is nameof(ETransport.kcp) or nameof(ETransport.xhttp)) | ||||||
|  |             { | ||||||
|  |                 errors.Add(string.Format(ResUI.CoreNotSupportNetwork, nameof(ECoreType.sing_box), net)); | ||||||
|  |                 return errors; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (item.ConfigType is not (EConfigType.VMess or EConfigType.VLESS or EConfigType.Trojan)) | ||||||
|  |             { | ||||||
|  |                 if (net is nameof(ETransport.ws) or nameof(ETransport.http) or nameof(ETransport.h2) or nameof(ETransport.quic) or nameof(ETransport.httpupgrade)) | ||||||
|  |                 { | ||||||
|  |                     errors.Add(string.Format(ResUI.CoreNotSupportProtocolTransport, nameof(ECoreType.sing_box), item.ConfigType.ToString(), net)); | ||||||
|  |                     return errors; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         else if (coreType is ECoreType.Xray) | ||||||
|  |         { | ||||||
|  |             // Xray core does not support these protocols | ||||||
|  |             if (item.ConfigType is EConfigType.Hysteria2 or EConfigType.TUIC or EConfigType.Anytls) | ||||||
|  |             { | ||||||
|  |                 errors.Add(string.Format(ResUI.CoreNotSupportProtocol, nameof(ECoreType.Xray), item.ConfigType.ToString())); | ||||||
|  |                 return errors; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return errors.Select(s => $"{item.Remarks}: {s}").ToList(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<List<string>> ValidateRelatedNodesExistAndValid(ProfileItem? item) | ||||||
|  |     { | ||||||
|  |         var errors = new List<string>(); | ||||||
|  |         errors.AddRange(await ValidateProxyChainedNodeExistAndValid(item)); | ||||||
|  |         errors.AddRange(await ValidateRoutingNodeExistAndValid(item)); | ||||||
|  |         return errors; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<List<string>> ValidateProxyChainedNodeExistAndValid(ProfileItem? item) | ||||||
|  |     { | ||||||
|  |         var errors = new List<string>(); | ||||||
|  |         if (item is null) | ||||||
|  |         { | ||||||
|  |             return errors; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // prev node and next node | ||||||
|  |         var subItem = await AppManager.Instance.GetSubItem(item.Subid); | ||||||
|  |         if (subItem is null) | ||||||
|  |         { | ||||||
|  |             return errors; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var prevNode = await AppManager.Instance.GetProfileItemViaRemarks(subItem.PrevProfile); | ||||||
|  |         var nextNode = await AppManager.Instance.GetProfileItemViaRemarks(subItem.NextProfile); | ||||||
|  |         var coreType = AppManager.Instance.GetCoreType(item, item.ConfigType); | ||||||
|  | 
 | ||||||
|  |         await CollectProxyChainedNodeValidation(prevNode, subItem.PrevProfile, coreType, errors); | ||||||
|  |         await CollectProxyChainedNodeValidation(nextNode, subItem.NextProfile, coreType, errors); | ||||||
|  | 
 | ||||||
|  |         return errors; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task CollectProxyChainedNodeValidation(ProfileItem? node, string tag, ECoreType coreType, List<string> errors) | ||||||
|  |     { | ||||||
|  |         if (node is not null) | ||||||
|  |         { | ||||||
|  |             var nodeErrors = await ValidateNodeAndCoreSupport(node, coreType); | ||||||
|  |             errors.AddRange(nodeErrors.Select(s => ResUI.ProxyChainedPrefix + s)); | ||||||
|  |         } | ||||||
|  |         else if (tag.IsNotEmpty()) | ||||||
|  |         { | ||||||
|  |             errors.Add(ResUI.ProxyChainedPrefix + string.Format(ResUI.NodeTagNotExist, tag)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<List<string>> ValidateRoutingNodeExistAndValid(ProfileItem? item) | ||||||
|  |     { | ||||||
|  |         var errors = new List<string>(); | ||||||
|  | 
 | ||||||
|  |         if (item is null) | ||||||
|  |         { | ||||||
|  |             return errors; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var coreType = AppManager.Instance.GetCoreType(item, item.ConfigType); | ||||||
|  |         var routing = await ConfigHandler.GetDefaultRouting(_config); | ||||||
|  |         if (routing == null) | ||||||
|  |         { | ||||||
|  |             return errors; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var rules = JsonUtils.Deserialize<List<RulesItem>>(routing.RuleSet); | ||||||
|  |         foreach (var ruleItem in rules ?? []) | ||||||
|  |         { | ||||||
|  |             if (!ruleItem.Enabled) | ||||||
|  |             { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var outboundTag = ruleItem.OutboundTag; | ||||||
|  |             if (outboundTag.IsNullOrEmpty() || Global.OutboundTags.Contains(outboundTag)) | ||||||
|  |             { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var tagItem = await AppManager.Instance.GetProfileItemViaRemarks(outboundTag); | ||||||
|  |             if (tagItem is null) | ||||||
|  |             { | ||||||
|  |                 errors.Add(ResUI.RoutingRuleOutboundPrefix + string.Format(ResUI.NodeTagNotExist, outboundTag)); | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var tagErrors = await ValidateNodeAndCoreSupport(tagItem, coreType); | ||||||
|  |             errors.AddRange(tagErrors.Select(s => ResUI.RoutingRuleOutboundPrefix + s)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return errors; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -15,35 +15,8 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             if (node?.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) |  | ||||||
|             { |  | ||||||
|                 ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); |  | ||||||
|                 if (profileGroupItem == null || profileGroupItem.ChildItems.IsNullOrEmpty()) |  | ||||||
|                 { |  | ||||||
|                     ret.Msg = ResUI.CheckServerSettings; |  | ||||||
|                     return ret; |  | ||||||
|                 } |  | ||||||
|                 var childProfiles = (await Task.WhenAll( |  | ||||||
|                         Utils.String2List(profileGroupItem.ChildItems) |  | ||||||
|                         .Where(p => !p.IsNullOrEmpty()) |  | ||||||
|                         .Select(AppManager.Instance.GetProfileItem) |  | ||||||
|                     )).Where(p => p != null).ToList(); |  | ||||||
|                 if (childProfiles.Count <= 0) |  | ||||||
|                 { |  | ||||||
|                     ret.Msg = ResUI.CheckServerSettings; |  | ||||||
|                     return ret; |  | ||||||
|                 } |  | ||||||
|                 switch (node.ConfigType) |  | ||||||
|                 { |  | ||||||
|                     case EConfigType.PolicyGroup: |  | ||||||
|                         return await GenerateClientMultipleLoadConfig(childProfiles, profileGroupItem.MultipleLoad); |  | ||||||
|                     case EConfigType.ProxyChain: |  | ||||||
|                         return await GenerateClientChainConfig(childProfiles); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (node == null |             if (node == null | ||||||
|                 || node.Port <= 0) |                 || !node.IsValid()) | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.CheckServerSettings; |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|                 return ret; |                 return ret; | ||||||
|  | @ -56,6 +29,17 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
| 
 | 
 | ||||||
|             ret.Msg = ResUI.InitialConfiguration; |             ret.Msg = ResUI.InitialConfiguration; | ||||||
|              |              | ||||||
|  |             if (node?.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |             { | ||||||
|  |                 switch (node.ConfigType) | ||||||
|  |                 { | ||||||
|  |                     case EConfigType.PolicyGroup: | ||||||
|  |                         return await GenerateClientMultipleLoadConfig(node); | ||||||
|  |                     case EConfigType.ProxyChain: | ||||||
|  |                         return await GenerateClientChainConfig(node); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             var result = EmbedUtils.GetEmbedText(Global.SingboxSampleClient); |             var result = EmbedUtils.GetEmbedText(Global.SingboxSampleClient); | ||||||
|             if (result.IsNullOrEmpty()) |             if (result.IsNullOrEmpty()) | ||||||
|             { |             { | ||||||
|  | @ -169,13 +153,10 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |                 if (item is null || item.IsComplex() || !item.IsValid()) | ||||||
|                 { |  | ||||||
|                     if (item is null || item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |  | ||||||
|                 { |                 { | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
|                 } |  | ||||||
| 
 | 
 | ||||||
|                 //find unused port |                 //find unused port | ||||||
|                 var port = initPort; |                 var port = initPort; | ||||||
|  | @ -214,27 +195,6 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|                 singboxConfig.inbounds.Add(inbound); |                 singboxConfig.inbounds.Add(inbound); | ||||||
| 
 | 
 | ||||||
|                 //outbound |                 //outbound | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                     && !Global.SsSecuritiesInSingbox.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS |  | ||||||
|                  && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VLESS or EConfigType.Trojan |  | ||||||
|                     && item.StreamSecurity == Global.StreamSecurityReality |  | ||||||
|                     && item.PublicKey.IsNullOrEmpty()) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 var server = await GenServer(item); |                 var server = await GenServer(item); | ||||||
|                 if (server is null) |                 if (server is null) | ||||||
|                 { |                 { | ||||||
|  | @ -293,7 +253,8 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             if (node is not { Port: > 0 }) |             if (node == null | ||||||
|  |                 || !node.IsValid()) | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.CheckServerSettings; |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|                 return ret; |                 return ret; | ||||||
|  | @ -371,7 +332,7 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task<RetResult> GenerateClientMultipleLoadConfig(List<ProfileItem> selecteds, EMultipleLoad multipleLoad) |     public async Task<RetResult> GenerateClientMultipleLoadConfig(ProfileItem parentNode) | ||||||
|     { |     { | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
|         try |         try | ||||||
|  | @ -405,53 +366,12 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|             await GenRouting(singboxConfig); |             await GenRouting(singboxConfig); | ||||||
|             await GenExperimental(singboxConfig); |             await GenExperimental(singboxConfig); | ||||||
| 
 | 
 | ||||||
|             var proxyProfiles = new List<ProfileItem>(); |             var groupRet = await GenGroupOutbound(parentNode, singboxConfig); | ||||||
|             foreach (var it in selecteds) |             if (groupRet != 0) | ||||||
|             { |  | ||||||
|                 if (it.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) |  | ||||||
|                 { |  | ||||||
|                     var itemGroup = await AppManager.Instance.GetProfileItem(it.IndexId); |  | ||||||
|                     proxyProfiles.Add(itemGroup); |  | ||||||
|                 } |  | ||||||
|                 if (!Global.SingboxSupportConfigType.Contains(it.ConfigType)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.Port <= 0) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |  | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |  | ||||||
|                 { |  | ||||||
|                     if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |  | ||||||
|                     { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                   && !Global.SsSecuritiesInSingbox.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 //outbound |  | ||||||
|                 proxyProfiles.Add(item); |  | ||||||
|             } |  | ||||||
|             if (proxyProfiles.Count <= 0) |  | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|                 return ret; |                 return ret; | ||||||
|             } |             } | ||||||
|             await GenOutboundsListWithChain(proxyProfiles, singboxConfig, multipleLoad); |  | ||||||
| 
 | 
 | ||||||
|             await GenDns(null, singboxConfig); |             await GenDns(null, singboxConfig); | ||||||
|             await ConvertGeo2Ruleset(singboxConfig); |             await ConvertGeo2Ruleset(singboxConfig); | ||||||
|  | @ -469,7 +389,7 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task<RetResult> GenerateClientChainConfig(List<ProfileItem> selecteds) |     public async Task<RetResult> GenerateClientChainConfig(ProfileItem parentNode) | ||||||
|     { |     { | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
|         try |         try | ||||||
|  | @ -503,48 +423,12 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|             await GenExperimental(singboxConfig); |             await GenExperimental(singboxConfig); | ||||||
|             singboxConfig.outbounds.RemoveAt(0); |             singboxConfig.outbounds.RemoveAt(0); | ||||||
| 
 | 
 | ||||||
|             var proxyProfiles = new List<ProfileItem>(); |             var groupRet = await GenGroupOutbound(parentNode, singboxConfig); | ||||||
|             foreach (var it in selecteds) |             if (groupRet != 0) | ||||||
|             { |  | ||||||
|                 if (!Global.SingboxSupportConfigType.Contains(it.ConfigType)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.Port <= 0) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |  | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |  | ||||||
|                 { |  | ||||||
|                     if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |  | ||||||
|                     { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                   && !Global.SsSecuritiesInSingbox.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 //outbound |  | ||||||
|                 proxyProfiles.Add(item); |  | ||||||
|             } |  | ||||||
|             if (proxyProfiles.Count <= 0) |  | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|                 return ret; |                 return ret; | ||||||
|             } |             } | ||||||
|             await GenChainOutboundsList(proxyProfiles, singboxConfig); |  | ||||||
| 
 | 
 | ||||||
|             await GenDns(null, singboxConfig); |             await GenDns(null, singboxConfig); | ||||||
|             await ConvertGeo2Ruleset(singboxConfig); |             await ConvertGeo2Ruleset(singboxConfig); | ||||||
|  |  | ||||||
|  | @ -204,6 +204,67 @@ public partial class CoreConfigSingboxService | ||||||
|         return await Task.FromResult<BaseServer4Sbox?>(null); |         return await Task.FromResult<BaseServer4Sbox?>(null); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     private async Task<int> GenGroupOutbound(ProfileItem node, SingboxConfig singboxConfig, string baseTagName = Global.ProxyTag, bool ignoreOriginChain = false) | ||||||
|  |     { | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             if (node.ConfigType is not (EConfigType.PolicyGroup or EConfigType.ProxyChain)) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); | ||||||
|  |             if (profileGroupItem is null || profileGroupItem.ChildItems.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             // remove custom nodes | ||||||
|  |             // remove group nodes for proxy chain | ||||||
|  |             // avoid self-reference | ||||||
|  |             var childProfiles = (await Task.WhenAll( | ||||||
|  |                     Utils.String2List(profileGroupItem.ChildItems) | ||||||
|  |                         .Where(p => !p.IsNullOrEmpty()) | ||||||
|  |                         .Select(AppManager.Instance.GetProfileItem) | ||||||
|  |                 )) | ||||||
|  |                 .Where(p => | ||||||
|  |                     p != null | ||||||
|  |                     && p.IsValid() | ||||||
|  |                     && p.ConfigType != EConfigType.Custom | ||||||
|  |                     && (node.ConfigType == EConfigType.PolicyGroup || p.ConfigType < EConfigType.Group) | ||||||
|  |                     && p.IndexId != node.IndexId | ||||||
|  |                 ) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             if (childProfiles.Count <= 0) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             switch (node.ConfigType) | ||||||
|  |             { | ||||||
|  |                 case EConfigType.PolicyGroup: | ||||||
|  |                     if (ignoreOriginChain) | ||||||
|  |                     { | ||||||
|  |                         await GenOutboundsList(childProfiles, singboxConfig, profileGroupItem.MultipleLoad, baseTagName); | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         await GenOutboundsListWithChain(childProfiles, singboxConfig, profileGroupItem.MultipleLoad, baseTagName); | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     break; | ||||||
|  |                 case EConfigType.ProxyChain: | ||||||
|  |                     await GenChainOutboundsList(childProfiles, singboxConfig, baseTagName); | ||||||
|  |                     break; | ||||||
|  |                 default: | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         catch (Exception ex) | ||||||
|  |         { | ||||||
|  |             Logging.SaveLog(_tag, ex); | ||||||
|  |         } | ||||||
|  |         return await Task.FromResult(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     private async Task<int> GenOutboundMux(ProfileItem node, Outbound4Sbox outbound) |     private async Task<int> GenOutboundMux(ProfileItem node, Outbound4Sbox outbound) | ||||||
|     { |     { | ||||||
|         try |         try | ||||||
|  |  | ||||||
|  | @ -385,29 +385,8 @@ public partial class CoreConfigSingboxService | ||||||
| 
 | 
 | ||||||
|         if (node.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) |         if (node.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|         { |         { | ||||||
|             ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); |  | ||||||
|             if (profileGroupItem == null || profileGroupItem.ChildItems.IsNullOrEmpty()) |  | ||||||
|             { |  | ||||||
|                 return Global.ProxyTag; |  | ||||||
|             } |  | ||||||
|             var childProfiles = (await Task.WhenAll( |  | ||||||
|                     Utils.String2List(profileGroupItem.ChildItems) |  | ||||||
|                     .Where(p => !p.IsNullOrEmpty()) |  | ||||||
|                     .Select(AppManager.Instance.GetProfileItem) |  | ||||||
|                 )).Where(p => p != null).ToList(); |  | ||||||
|             if (childProfiles.Count <= 0) |  | ||||||
|             { |  | ||||||
|                 return Global.ProxyTag; |  | ||||||
|             } |  | ||||||
|             var childBaseTagName = $"{Global.ProxyTag}-{node.IndexId}"; |             var childBaseTagName = $"{Global.ProxyTag}-{node.IndexId}"; | ||||||
|             var ret = node.ConfigType switch |             var ret = await GenGroupOutbound(node, singboxConfig, childBaseTagName); | ||||||
|             { |  | ||||||
|                 EConfigType.PolicyGroup => |  | ||||||
|                     await GenOutboundsListWithChain(childProfiles, singboxConfig, profileGroupItem.MultipleLoad, childBaseTagName), |  | ||||||
|                 EConfigType.ProxyChain => |  | ||||||
|                     await GenChainOutboundsList(childProfiles, singboxConfig, childBaseTagName), |  | ||||||
|                 _ => throw new NotImplementedException() |  | ||||||
|             }; |  | ||||||
|             if (ret == 0) |             if (ret == 0) | ||||||
|             { |             { | ||||||
|                 return childBaseTagName; |                 return childBaseTagName; | ||||||
|  |  | ||||||
|  | @ -15,35 +15,8 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             if (node?.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) |  | ||||||
|             { |  | ||||||
|                 ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); |  | ||||||
|                 if (profileGroupItem == null || profileGroupItem.ChildItems.IsNullOrEmpty()) |  | ||||||
|                 { |  | ||||||
|                     ret.Msg = ResUI.CheckServerSettings; |  | ||||||
|                     return ret; |  | ||||||
|                 } |  | ||||||
|                 var childProfiles = (await Task.WhenAll( |  | ||||||
|                         Utils.String2List(profileGroupItem.ChildItems) |  | ||||||
|                         .Where(p => !p.IsNullOrEmpty()) |  | ||||||
|                         .Select(AppManager.Instance.GetProfileItem) |  | ||||||
|                     )).Where(p => p != null).ToList(); |  | ||||||
|                 if (childProfiles.Count <= 0) |  | ||||||
|                 { |  | ||||||
|                     ret.Msg = ResUI.CheckServerSettings; |  | ||||||
|                     return ret; |  | ||||||
|                 } |  | ||||||
|                 switch (node.ConfigType) |  | ||||||
|                 { |  | ||||||
|                     case EConfigType.PolicyGroup: |  | ||||||
|                         return await GenerateClientMultipleLoadConfig(childProfiles, profileGroupItem.MultipleLoad); |  | ||||||
|                     case EConfigType.ProxyChain: |  | ||||||
|                         return await GenerateClientChainConfig(childProfiles); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (node == null |             if (node == null | ||||||
|                 || node.Port <= 0) |                 || !node.IsValid()) | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.CheckServerSettings; |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|                 return ret; |                 return ret; | ||||||
|  | @ -57,6 +30,17 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
| 
 | 
 | ||||||
|             ret.Msg = ResUI.InitialConfiguration; |             ret.Msg = ResUI.InitialConfiguration; | ||||||
| 
 | 
 | ||||||
|  |             if (node?.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |             { | ||||||
|  |                 switch (node.ConfigType) | ||||||
|  |                 { | ||||||
|  |                     case EConfigType.PolicyGroup: | ||||||
|  |                         return await GenerateClientMultipleLoadConfig(node); | ||||||
|  |                     case EConfigType.ProxyChain: | ||||||
|  |                         return await GenerateClientChainConfig(node); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             var result = EmbedUtils.GetEmbedText(Global.V2raySampleClient); |             var result = EmbedUtils.GetEmbedText(Global.V2raySampleClient); | ||||||
|             if (result.IsNullOrEmpty()) |             if (result.IsNullOrEmpty()) | ||||||
|             { |             { | ||||||
|  | @ -98,7 +82,7 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task<RetResult> GenerateClientMultipleLoadConfig(List<ProfileItem> selecteds, EMultipleLoad multipleLoad) |     public async Task<RetResult> GenerateClientMultipleLoadConfig(ProfileItem parentNode) | ||||||
|     { |     { | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
| 
 | 
 | ||||||
|  | @ -134,57 +118,12 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|             await GenDns(null, v2rayConfig); |             await GenDns(null, v2rayConfig); | ||||||
|             await GenStatistic(v2rayConfig); |             await GenStatistic(v2rayConfig); | ||||||
| 
 | 
 | ||||||
|             var proxyProfiles = new List<ProfileItem>(); |             var groupRet = await GenGroupOutbound(parentNode, v2rayConfig); | ||||||
|             foreach (var it in selecteds) |             if (groupRet != 0) | ||||||
|             { |  | ||||||
|                 if (it.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) |  | ||||||
|                 { |  | ||||||
|                     var itemGroup = await AppManager.Instance.GetProfileItem(it.IndexId); |  | ||||||
|                     proxyProfiles.Add(itemGroup); |  | ||||||
|                 } |  | ||||||
|                 if (!Global.XraySupportConfigType.Contains(it.ConfigType)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.Port <= 0) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |  | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |  | ||||||
|                 { |  | ||||||
|                     if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |  | ||||||
|                     { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                   && !Global.SsSecuritiesInSingbox.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 //outbound |  | ||||||
|                 proxyProfiles.Add(item); |  | ||||||
|             } |  | ||||||
|             if (proxyProfiles.Count <= 0) |  | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|                 return ret; |                 return ret; | ||||||
|             } |             } | ||||||
|             await GenOutboundsListWithChain(proxyProfiles, v2rayConfig); |  | ||||||
| 
 |  | ||||||
|             //add balancers |  | ||||||
|             await GenObservatory(v2rayConfig, multipleLoad); |  | ||||||
|             await GenBalancer(v2rayConfig, multipleLoad); |  | ||||||
| 
 | 
 | ||||||
|             var defaultBalancerTag = $"{Global.ProxyTag}{Global.BalancerTagSuffix}"; |             var defaultBalancerTag = $"{Global.ProxyTag}{Global.BalancerTagSuffix}"; | ||||||
| 
 | 
 | ||||||
|  | @ -248,7 +187,7 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task<RetResult> GenerateClientChainConfig(List<ProfileItem> selecteds) |     public async Task<RetResult> GenerateClientChainConfig(ProfileItem parentNode) | ||||||
|     { |     { | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
| 
 | 
 | ||||||
|  | @ -284,48 +223,12 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|             await GenStatistic(v2rayConfig); |             await GenStatistic(v2rayConfig); | ||||||
|             v2rayConfig.outbounds.RemoveAt(0); |             v2rayConfig.outbounds.RemoveAt(0); | ||||||
| 
 | 
 | ||||||
|             var proxyProfiles = new List<ProfileItem>(); |             var groupRet = await GenGroupOutbound(parentNode, v2rayConfig); | ||||||
|             foreach (var it in selecteds) |             if (groupRet != 0) | ||||||
|             { |  | ||||||
|                 if (!Global.XraySupportConfigType.Contains(it.ConfigType)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.Port <= 0) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |  | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |  | ||||||
|                 { |  | ||||||
|                     if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |  | ||||||
|                     { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                   && !Global.SsSecuritiesInSingbox.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 //outbound |  | ||||||
|                 proxyProfiles.Add(item); |  | ||||||
|             } |  | ||||||
|             if (proxyProfiles.Count <= 0) |  | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|                 return ret; |                 return ret; | ||||||
|             } |             } | ||||||
|             await GenChainOutboundsList(proxyProfiles, v2rayConfig); |  | ||||||
| 
 | 
 | ||||||
|             ret.Success = true; |             ret.Success = true; | ||||||
| 
 | 
 | ||||||
|  | @ -398,13 +301,10 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |                 if (item is null || item.IsComplex() || !item.IsValid()) | ||||||
|                 { |  | ||||||
|                     if (item is null || item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |  | ||||||
|                 { |                 { | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
|                 } |  | ||||||
| 
 | 
 | ||||||
|                 //find unused port |                 //find unused port | ||||||
|                 var port = initPort; |                 var port = initPort; | ||||||
|  | @ -432,28 +332,6 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|                 it.Port = port; |                 it.Port = port; | ||||||
|                 it.AllowTest = true; |                 it.AllowTest = true; | ||||||
| 
 | 
 | ||||||
|                 //outbound |  | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                     && !Global.SsSecuritiesInXray.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS |  | ||||||
|                  && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VLESS or EConfigType.Trojan |  | ||||||
|                     && item.StreamSecurity == Global.StreamSecurityReality |  | ||||||
|                     && item.PublicKey.IsNullOrEmpty()) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 //inbound |                 //inbound | ||||||
|                 Inbounds4Ray inbound = new() |                 Inbounds4Ray inbound = new() | ||||||
|                 { |                 { | ||||||
|  | @ -464,6 +342,7 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|                 inbound.tag = inbound.protocol + inbound.port.ToString(); |                 inbound.tag = inbound.protocol + inbound.port.ToString(); | ||||||
|                 v2rayConfig.inbounds.Add(inbound); |                 v2rayConfig.inbounds.Add(inbound); | ||||||
| 
 | 
 | ||||||
|  |                 //outbound | ||||||
|                 var outbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); |                 var outbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); | ||||||
|                 await GenOutbound(item, outbound); |                 await GenOutbound(item, outbound); | ||||||
|                 outbound.tag = Global.ProxyTag + inbound.port.ToString(); |                 outbound.tag = Global.ProxyTag + inbound.port.ToString(); | ||||||
|  | @ -497,7 +376,8 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             if (node is not { Port: > 0 }) |             if (node == null | ||||||
|  |                 || !node.IsValid()) | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.CheckServerSettings; |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|                 return ret; |                 return ret; | ||||||
|  |  | ||||||
|  | @ -480,6 +480,69 @@ public partial class CoreConfigV2rayService | ||||||
|         return 0; |         return 0; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     private async Task<int> GenGroupOutbound(ProfileItem node, V2rayConfig v2rayConfig, string baseTagName = Global.ProxyTag, bool ignoreOriginChain = false) | ||||||
|  |     { | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             if (node.ConfigType is not (EConfigType.PolicyGroup or EConfigType.ProxyChain)) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); | ||||||
|  |             if (profileGroupItem is null || profileGroupItem.ChildItems.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             // remove custom nodes | ||||||
|  |             // remove group nodes for proxy chain | ||||||
|  |             var childProfiles = (await Task.WhenAll( | ||||||
|  |                     Utils.String2List(profileGroupItem.ChildItems) | ||||||
|  |                         .Where(p => !p.IsNullOrEmpty()) | ||||||
|  |                         .Select(AppManager.Instance.GetProfileItem) | ||||||
|  |                 )) | ||||||
|  |                 .Where(p => | ||||||
|  |                     p != null && | ||||||
|  |                     p.IsValid() && | ||||||
|  |                     p.ConfigType != EConfigType.Custom && | ||||||
|  |                     (node.ConfigType == EConfigType.PolicyGroup || p.ConfigType < EConfigType.Group) | ||||||
|  |                 ) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             if (childProfiles.Count <= 0) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             switch (node.ConfigType) | ||||||
|  |             { | ||||||
|  |                 case EConfigType.PolicyGroup: | ||||||
|  |                     if (ignoreOriginChain) | ||||||
|  |                     { | ||||||
|  |                         await GenOutboundsList(childProfiles, v2rayConfig, baseTagName); | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         await GenOutboundsListWithChain(childProfiles, v2rayConfig, baseTagName); | ||||||
|  |                     } | ||||||
|  |                     break; | ||||||
|  |                 case EConfigType.ProxyChain: | ||||||
|  |                     await GenChainOutboundsList(childProfiles, v2rayConfig, baseTagName); | ||||||
|  |                     break; | ||||||
|  |                 default: | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             //add balancers | ||||||
|  |             await GenObservatory(v2rayConfig, profileGroupItem.MultipleLoad, baseTagName); | ||||||
|  |             await GenBalancer(v2rayConfig, profileGroupItem.MultipleLoad, baseTagName); | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  |         catch (Exception ex) | ||||||
|  |         { | ||||||
|  |             Logging.SaveLog(_tag, ex); | ||||||
|  |         } | ||||||
|  |         return await Task.FromResult(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     private async Task<int> GenMoreOutbounds(ProfileItem node, V2rayConfig v2rayConfig) |     private async Task<int> GenMoreOutbounds(ProfileItem node, V2rayConfig v2rayConfig) | ||||||
|     { |     { | ||||||
|         //fragment proxy |         //fragment proxy | ||||||
|  |  | ||||||
|  | @ -141,33 +141,8 @@ public partial class CoreConfigV2rayService | ||||||
| 
 | 
 | ||||||
|         if (node.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) |         if (node.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|         { |         { | ||||||
|             ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); |  | ||||||
|             if (profileGroupItem == null || profileGroupItem.ChildItems.IsNullOrEmpty()) |  | ||||||
|             { |  | ||||||
|                 return Global.ProxyTag; |  | ||||||
|             } |  | ||||||
|             var childProfiles = (await Task.WhenAll( |  | ||||||
|                     Utils.String2List(profileGroupItem.ChildItems) |  | ||||||
|                     .Where(p => !p.IsNullOrEmpty()) |  | ||||||
|                     .Select(AppManager.Instance.GetProfileItem) |  | ||||||
|                 )).Where(p => p != null).ToList(); |  | ||||||
|             if (childProfiles.Count <= 0) |  | ||||||
|             { |  | ||||||
|                 return Global.ProxyTag; |  | ||||||
|             } |  | ||||||
|             var childBaseTagName = $"{Global.ProxyTag}-{node.IndexId}"; |             var childBaseTagName = $"{Global.ProxyTag}-{node.IndexId}"; | ||||||
|             var ret = node.ConfigType switch |             var ret = await GenGroupOutbound(node, v2rayConfig, childBaseTagName); | ||||||
|             { |  | ||||||
|                 EConfigType.PolicyGroup => |  | ||||||
|                     await GenOutboundsListWithChain(childProfiles, v2rayConfig, childBaseTagName), |  | ||||||
|                 EConfigType.ProxyChain => |  | ||||||
|                     await GenChainOutboundsList(childProfiles, v2rayConfig, childBaseTagName), |  | ||||||
|                 _ => throw new NotImplementedException() |  | ||||||
|             }; |  | ||||||
|             if (node.ConfigType == EConfigType.PolicyGroup) |  | ||||||
|             { |  | ||||||
|                 await GenBalancer(v2rayConfig, profileGroupItem.MultipleLoad, childBaseTagName); |  | ||||||
|             } |  | ||||||
|             if (ret == 0) |             if (ret == 0) | ||||||
|             { |             { | ||||||
|                 return childBaseTagName; |                 return childBaseTagName; | ||||||
|  |  | ||||||
|  | @ -69,6 +69,8 @@ public class ClashProxiesViewModel : MyReactiveObject | ||||||
|         SortingSelected = _config.ClashUIItem.ProxiesSorting; |         SortingSelected = _config.ClashUIItem.ProxiesSorting; | ||||||
|         RuleModeSelected = (int)_config.ClashUIItem.RuleMode; |         RuleModeSelected = (int)_config.ClashUIItem.RuleMode; | ||||||
| 
 | 
 | ||||||
|  |         #region WhenAnyValue && ReactiveCommand | ||||||
|  | 
 | ||||||
|         this.WhenAnyValue( |         this.WhenAnyValue( | ||||||
|            x => x.SelectedGroup, |            x => x.SelectedGroup, | ||||||
|            y => y != null && y.Name.IsNotEmpty()) |            y => y != null && y.Name.IsNotEmpty()) | ||||||
|  | @ -89,6 +91,17 @@ public class ClashProxiesViewModel : MyReactiveObject | ||||||
|         y => y == true) |         y => y == true) | ||||||
|             .Subscribe(c => { _config.ClashUIItem.ProxiesAutoRefresh = AutoRefresh; }); |             .Subscribe(c => { _config.ClashUIItem.ProxiesAutoRefresh = AutoRefresh; }); | ||||||
| 
 | 
 | ||||||
|  |         #endregion WhenAnyValue && ReactiveCommand | ||||||
|  | 
 | ||||||
|  |         #region AppEvents | ||||||
|  | 
 | ||||||
|  |         AppEvents.ProxiesReloadRequested | ||||||
|  |             .AsObservable() | ||||||
|  |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|  |             .Subscribe(async _ => await ProxiesReload()); | ||||||
|  | 
 | ||||||
|  |         #endregion AppEvents | ||||||
|  | 
 | ||||||
|         _ = Init(); |         _ = Init(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2,7 +2,6 @@ using System.Reactive; | ||||||
| using System.Reactive.Concurrency; | using System.Reactive.Concurrency; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using ReactiveUI.Fody.Helpers; | using ReactiveUI.Fody.Helpers; | ||||||
| using Splat; |  | ||||||
| 
 | 
 | ||||||
| namespace ServiceLib.ViewModels; | namespace ServiceLib.ViewModels; | ||||||
| 
 | 
 | ||||||
|  | @ -251,7 +250,6 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|         BlReloadEnabled = true; |         BlReloadEnabled = true; | ||||||
|         await Reload(); |         await Reload(); | ||||||
|         await AutoHideStartup(); |         await AutoHideStartup(); | ||||||
|         Locator.Current.GetService<StatusBarViewModel>()?.RefreshRoutingsMenu(); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #endregion Init |     #endregion Init | ||||||
|  | @ -312,7 +310,7 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
| 
 | 
 | ||||||
|     private void RefreshSubscriptions() |     private void RefreshSubscriptions() | ||||||
|     { |     { | ||||||
|         Locator.Current.GetService<ProfilesViewModel>()?.RefreshSubscriptions(); |         AppEvents.SubscriptionsRefreshRequested.OnNext(Unit.Default); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #endregion Servers && Groups |     #endregion Servers && Groups | ||||||
|  | @ -448,7 +446,7 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|         var ret = await _updateView?.Invoke(EViewAction.OptionSettingWindow, null); |         var ret = await _updateView?.Invoke(EViewAction.OptionSettingWindow, null); | ||||||
|         if (ret == true) |         if (ret == true) | ||||||
|         { |         { | ||||||
|             Locator.Current.GetService<StatusBarViewModel>()?.InboundDisplayStatus(); |             AppEvents.InboundDisplayRequested.OnNext(Unit.Default); | ||||||
|             await Reload(); |             await Reload(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -459,7 +457,7 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|         if (ret == true) |         if (ret == true) | ||||||
|         { |         { | ||||||
|             await ConfigHandler.InitBuiltinRouting(_config); |             await ConfigHandler.InitBuiltinRouting(_config); | ||||||
|             Locator.Current.GetService<StatusBarViewModel>()?.RefreshRoutingsMenu(); |             AppEvents.RoutingsMenuRefreshRequested.OnNext(Unit.Default); | ||||||
|             await Reload(); |             await Reload(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -533,9 +531,15 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|             await SysProxyHandler.UpdateSysProxy(_config, false); |             await SysProxyHandler.UpdateSysProxy(_config, false); | ||||||
|             await Task.Delay(1000); |             await Task.Delay(1000); | ||||||
|         }); |         }); | ||||||
|         Locator.Current.GetService<StatusBarViewModel>()?.TestServerAvailability(); |         AppEvents.TestServerRequested.OnNext(Unit.Default); | ||||||
| 
 | 
 | ||||||
|         RxApp.MainThreadScheduler.Schedule(() => _ = ReloadResult()); |         var showClashUI = _config.IsRunningCore(ECoreType.sing_box); | ||||||
|  |         if (showClashUI) | ||||||
|  |         { | ||||||
|  |             AppEvents.ProxiesReloadRequested.OnNext(Unit.Default); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         RxApp.MainThreadScheduler.Schedule(() => ReloadResult(showClashUI)); | ||||||
| 
 | 
 | ||||||
|         BlReloadEnabled = true; |         BlReloadEnabled = true; | ||||||
|         if (_hasNextReloadJob) |         if (_hasNextReloadJob) | ||||||
|  | @ -545,19 +549,11 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task ReloadResult() |     private void ReloadResult(bool showClashUI) | ||||||
|     { |     { | ||||||
|         // BlReloadEnabled = true; |         // BlReloadEnabled = true; | ||||||
|         //Locator.Current.GetService<StatusBarViewModel>()?.ChangeSystemProxyAsync(_config.systemProxyItem.sysProxyType, false); |         ShowClashUI = showClashUI; | ||||||
|         ShowClashUI = _config.IsRunningCore(ECoreType.sing_box); |         TabMainSelectedIndex = showClashUI ? TabMainSelectedIndex : 0; | ||||||
|         if (ShowClashUI) |  | ||||||
|         { |  | ||||||
|             Locator.Current.GetService<ClashProxiesViewModel>()?.ProxiesReload(); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             TabMainSelectedIndex = 0; |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async Task LoadCore() |     private async Task LoadCore() | ||||||
|  | @ -589,7 +585,7 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|     { |     { | ||||||
|         await ConfigHandler.ApplyRegionalPreset(_config, type); |         await ConfigHandler.ApplyRegionalPreset(_config, type); | ||||||
|         await ConfigHandler.InitRouting(_config); |         await ConfigHandler.InitRouting(_config); | ||||||
|         Locator.Current.GetService<StatusBarViewModel>()?.RefreshRoutingsMenu(); |         AppEvents.RoutingsMenuRefreshRequested.OnNext(Unit.Default); | ||||||
| 
 | 
 | ||||||
|         await ConfigHandler.SaveConfig(_config); |         await ConfigHandler.SaveConfig(_config); | ||||||
|         await new UpdateService().UpdateGeoFileAll(_config, UpdateTaskHandler); |         await new UpdateService().UpdateGeoFileAll(_config, UpdateTaskHandler); | ||||||
|  |  | ||||||
|  | @ -250,11 +250,21 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|             .ObserveOn(RxApp.MainThreadScheduler) |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|             .Subscribe(async _ => await RefreshServersBiz()); |             .Subscribe(async _ => await RefreshServersBiz()); | ||||||
| 
 | 
 | ||||||
|  |         AppEvents.SubscriptionsRefreshRequested | ||||||
|  |             .AsObservable() | ||||||
|  |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|  |             .Subscribe(async _ => await RefreshSubscriptions()); | ||||||
|  | 
 | ||||||
|         AppEvents.DispatcherStatisticsRequested |         AppEvents.DispatcherStatisticsRequested | ||||||
|             .AsObservable() |             .AsObservable() | ||||||
|             .ObserveOn(RxApp.MainThreadScheduler) |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|             .Subscribe(async result => await UpdateStatistics(result)); |             .Subscribe(async result => await UpdateStatistics(result)); | ||||||
| 
 | 
 | ||||||
|  |         AppEvents.SetDefaultServerRequested | ||||||
|  |             .AsObservable() | ||||||
|  |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|  |             .Subscribe(async indexId => await SetDefaultServer(indexId)); | ||||||
|  | 
 | ||||||
|         #endregion AppEvents |         #endregion AppEvents | ||||||
| 
 | 
 | ||||||
|         _ = Init(); |         _ = Init(); | ||||||
|  | @ -390,7 +400,7 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|         await _updateView?.Invoke(EViewAction.DispatcherRefreshServersBiz, null); |         await _updateView?.Invoke(EViewAction.DispatcherRefreshServersBiz, null); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task RefreshSubscriptions() |     private async Task RefreshSubscriptions() | ||||||
|     { |     { | ||||||
|         SubItems.Clear(); |         SubItems.Clear(); | ||||||
| 
 | 
 | ||||||
|  | @ -579,7 +589,7 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|         await SetDefaultServer(SelectedProfile.IndexId); |         await SetDefaultServer(SelectedProfile.IndexId); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task SetDefaultServer(string? indexId) |     private async Task SetDefaultServer(string? indexId) | ||||||
|     { |     { | ||||||
|         if (indexId.IsNullOrEmpty()) |         if (indexId.IsNullOrEmpty()) | ||||||
|         { |         { | ||||||
|  | @ -596,6 +606,16 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         var msgs = await ActionPrecheckService.Instance.CheckBeforeSetActive(indexId); | ||||||
|  |         foreach (var msg in msgs) | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.SendMessage(msg); | ||||||
|  |         } | ||||||
|  |         if (msgs.Count > 0) | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.Enqueue(msgs.First()); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         if (await ConfigHandler.SetDefaultServerIndex(_config, indexId) == 0) |         if (await ConfigHandler.SetDefaultServerIndex(_config, indexId) == 0) | ||||||
|         { |         { | ||||||
|             await RefreshServers(); |             await RefreshServers(); | ||||||
|  | @ -759,6 +779,16 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|             NoticeManager.Instance.Enqueue(ResUI.PleaseSelectServer); |             NoticeManager.Instance.Enqueue(ResUI.PleaseSelectServer); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         var msgs = await ActionPrecheckService.Instance.CheckBeforeGenerateConfig(item); | ||||||
|  |         foreach (var msg in msgs) | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.SendMessage(msg); | ||||||
|  |         } | ||||||
|  |         if (msgs.Count > 0) | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.Enqueue(msgs.First()); | ||||||
|  |         } | ||||||
|         if (blClipboard) |         if (blClipboard) | ||||||
|         { |         { | ||||||
|             var result = await CoreConfigHandler.GenerateClientConfig(item, null); |             var result = await CoreConfigHandler.GenerateClientConfig(item, null); | ||||||
|  |  | ||||||
|  | @ -11,6 +11,9 @@ namespace ServiceLib.ViewModels; | ||||||
| 
 | 
 | ||||||
| public class StatusBarViewModel : MyReactiveObject | public class StatusBarViewModel : MyReactiveObject | ||||||
| { | { | ||||||
|  |     private static readonly Lazy<StatusBarViewModel> _instance = new(() => new(null)); | ||||||
|  |     public static StatusBarViewModel Instance => _instance.Value; | ||||||
|  | 
 | ||||||
|     #region ObservableCollection |     #region ObservableCollection | ||||||
| 
 | 
 | ||||||
|     public IObservableCollection<RoutingItem> RoutingItems { get; } = new ObservableCollectionExtended<RoutingItem>(); |     public IObservableCollection<RoutingItem> RoutingItems { get; } = new ObservableCollectionExtended<RoutingItem>(); | ||||||
|  | @ -209,6 +212,26 @@ public class StatusBarViewModel : MyReactiveObject | ||||||
|             .ObserveOn(RxApp.MainThreadScheduler) |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|             .Subscribe(async result => await UpdateStatistics(result)); |             .Subscribe(async result => await UpdateStatistics(result)); | ||||||
| 
 | 
 | ||||||
|  |         AppEvents.RoutingsMenuRefreshRequested | ||||||
|  |             .AsObservable() | ||||||
|  |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|  |             .Subscribe(async _ => await RefreshRoutingsMenu()); | ||||||
|  | 
 | ||||||
|  |         AppEvents.TestServerRequested | ||||||
|  |             .AsObservable() | ||||||
|  |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|  |             .Subscribe(async _ => await TestServerAvailability()); | ||||||
|  | 
 | ||||||
|  |         AppEvents.InboundDisplayRequested | ||||||
|  |             .AsObservable() | ||||||
|  |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|  |             .Subscribe(async _ => await InboundDisplayStatus()); | ||||||
|  | 
 | ||||||
|  |         AppEvents.SysProxyChangeRequested | ||||||
|  |             .AsObservable() | ||||||
|  |             .ObserveOn(RxApp.MainThreadScheduler) | ||||||
|  |             .Subscribe(async result => await SetListenerType(result)); | ||||||
|  | 
 | ||||||
|         #endregion AppEvents |         #endregion AppEvents | ||||||
| 
 | 
 | ||||||
|         _ = Init(); |         _ = Init(); | ||||||
|  | @ -329,7 +352,7 @@ public class StatusBarViewModel : MyReactiveObject | ||||||
|         { |         { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Locator.Current.GetService<ProfilesViewModel>()?.SetDefaultServer(SelectedServer.ID); |         AppEvents.SetDefaultServerRequested.OnNext(SelectedServer.ID); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task TestServerAvailability() |     public async Task TestServerAvailability() | ||||||
|  | @ -364,7 +387,7 @@ public class StatusBarViewModel : MyReactiveObject | ||||||
| 
 | 
 | ||||||
|     #region System proxy and Routings |     #region System proxy and Routings | ||||||
| 
 | 
 | ||||||
|     public async Task SetListenerType(ESysProxyType type) |     private async Task SetListenerType(ESysProxyType type) | ||||||
|     { |     { | ||||||
|         if (_config.SystemProxyItem.SysProxyType == type) |         if (_config.SystemProxyItem.SysProxyType == type) | ||||||
|         { |         { | ||||||
|  | @ -393,7 +416,7 @@ public class StatusBarViewModel : MyReactiveObject | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task RefreshRoutingsMenu() |     private async Task RefreshRoutingsMenu() | ||||||
|     { |     { | ||||||
|         RoutingItems.Clear(); |         RoutingItems.Clear(); | ||||||
| 
 | 
 | ||||||
|  | @ -501,7 +524,7 @@ public class StatusBarViewModel : MyReactiveObject | ||||||
| 
 | 
 | ||||||
|     #region UI |     #region UI | ||||||
| 
 | 
 | ||||||
|     public async Task InboundDisplayStatus() |     private async Task InboundDisplayStatus() | ||||||
|     { |     { | ||||||
|         StringBuilder sb = new(); |         StringBuilder sb = new(); | ||||||
|         sb.Append($"[{EInboundProtocol.mixed}:{AppManager.Instance.GetLocalPort(EInboundProtocol.socks)}"); |         sb.Append($"[{EInboundProtocol.mixed}:{AppManager.Instance.GetLocalPort(EInboundProtocol.socks)}"); | ||||||
|  |  | ||||||
|  | @ -16,9 +16,7 @@ public partial class App : Application | ||||||
|         AppDomain.CurrentDomain.UnhandledException += CurrentDomain_UnhandledException; |         AppDomain.CurrentDomain.UnhandledException += CurrentDomain_UnhandledException; | ||||||
|         TaskScheduler.UnobservedTaskException += TaskScheduler_UnobservedTaskException; |         TaskScheduler.UnobservedTaskException += TaskScheduler_UnobservedTaskException; | ||||||
| 
 | 
 | ||||||
|         var ViewModel = new StatusBarViewModel(null); |         DataContext = StatusBarViewModel.Instance; | ||||||
|         Locator.CurrentMutable.RegisterLazySingleton(() => ViewModel, typeof(StatusBarViewModel)); |  | ||||||
|         DataContext = ViewModel; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public override void OnFrameworkInitializationCompleted() |     public override void OnFrameworkInitializationCompleted() | ||||||
|  |  | ||||||
|  | @ -3,7 +3,6 @@ using Avalonia.Input; | ||||||
| using Avalonia.ReactiveUI; | using Avalonia.ReactiveUI; | ||||||
| using DynamicData; | using DynamicData; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using Splat; |  | ||||||
| 
 | 
 | ||||||
| namespace v2rayN.Desktop.Views; | namespace v2rayN.Desktop.Views; | ||||||
| 
 | 
 | ||||||
|  | @ -13,7 +12,6 @@ public partial class ClashProxiesView : ReactiveUserControl<ClashProxiesViewMode | ||||||
|     { |     { | ||||||
|         InitializeComponent(); |         InitializeComponent(); | ||||||
|         ViewModel = new ClashProxiesViewModel(UpdateViewHandler); |         ViewModel = new ClashProxiesViewModel(UpdateViewHandler); | ||||||
|         Locator.CurrentMutable.RegisterLazySingleton(() => ViewModel, typeof(ClashProxiesViewModel)); |  | ||||||
|         lstProxyDetails.DoubleTapped += LstProxyDetails_DoubleTapped; |         lstProxyDetails.DoubleTapped += LstProxyDetails_DoubleTapped; | ||||||
|         this.KeyDown += ClashProxiesView_KeyDown; |         this.KeyDown += ClashProxiesView_KeyDown; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -20,7 +20,7 @@ namespace v2rayN.Desktop.Views; | ||||||
| public partial class MainWindow : WindowBase<MainWindowViewModel> | public partial class MainWindow : WindowBase<MainWindowViewModel> | ||||||
| { | { | ||||||
|     private static Config _config; |     private static Config _config; | ||||||
|     private WindowNotificationManager? _manager; |     private readonly WindowNotificationManager? _manager; | ||||||
|     private CheckUpdateView? _checkUpdateView; |     private CheckUpdateView? _checkUpdateView; | ||||||
|     private BackupAndRestoreView? _backupAndRestoreView; |     private BackupAndRestoreView? _backupAndRestoreView; | ||||||
|     private bool _blCloseByUser = false; |     private bool _blCloseByUser = false; | ||||||
|  | @ -266,7 +266,7 @@ public partial class MainWindow : WindowBase<MainWindowViewModel> | ||||||
|             case EGlobalHotkey.SystemProxySet: |             case EGlobalHotkey.SystemProxySet: | ||||||
|             case EGlobalHotkey.SystemProxyUnchanged: |             case EGlobalHotkey.SystemProxyUnchanged: | ||||||
|             case EGlobalHotkey.SystemProxyPac: |             case EGlobalHotkey.SystemProxyPac: | ||||||
|                 Locator.Current.GetService<StatusBarViewModel>()?.SetListenerType((ESysProxyType)((int)e - 1)); |                 AppEvents.SysProxyChangeRequested.OnNext((ESysProxyType)((int)e - 1)); | ||||||
|                 break; |                 break; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -8,7 +8,6 @@ using Avalonia.Threading; | ||||||
| using DialogHostAvalonia; | using DialogHostAvalonia; | ||||||
| using MsBox.Avalonia.Enums; | using MsBox.Avalonia.Enums; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using Splat; |  | ||||||
| using v2rayN.Desktop.Common; | using v2rayN.Desktop.Common; | ||||||
| 
 | 
 | ||||||
| namespace v2rayN.Desktop.Views; | namespace v2rayN.Desktop.Views; | ||||||
|  | @ -48,7 +47,6 @@ public partial class ProfilesView : ReactiveUserControl<ProfilesViewModel> | ||||||
|         //} |         //} | ||||||
| 
 | 
 | ||||||
|         ViewModel = new ProfilesViewModel(UpdateViewHandler); |         ViewModel = new ProfilesViewModel(UpdateViewHandler); | ||||||
|         Locator.CurrentMutable.RegisterLazySingleton(() => ViewModel, typeof(ProfilesViewModel)); |  | ||||||
| 
 | 
 | ||||||
|         this.WhenActivated(disposables => |         this.WhenActivated(disposables => | ||||||
|         { |         { | ||||||
|  |  | ||||||
|  | @ -6,7 +6,6 @@ using Avalonia.ReactiveUI; | ||||||
| using Avalonia.Threading; | using Avalonia.Threading; | ||||||
| using DialogHostAvalonia; | using DialogHostAvalonia; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using Splat; |  | ||||||
| using v2rayN.Desktop.Common; | using v2rayN.Desktop.Common; | ||||||
| 
 | 
 | ||||||
| namespace v2rayN.Desktop.Views; | namespace v2rayN.Desktop.Views; | ||||||
|  | @ -20,9 +19,8 @@ public partial class StatusBarView : ReactiveUserControl<StatusBarViewModel> | ||||||
|         InitializeComponent(); |         InitializeComponent(); | ||||||
| 
 | 
 | ||||||
|         _config = AppManager.Instance.Config; |         _config = AppManager.Instance.Config; | ||||||
|         //ViewModel = new StatusBarViewModel(UpdateViewHandler); |          | ||||||
|         //Locator.CurrentMutable.RegisterLazySingleton(() => ViewModel, typeof(StatusBarViewModel)); |         ViewModel = StatusBarViewModel.Instance; | ||||||
|         ViewModel = Locator.Current.GetService<StatusBarViewModel>(); |  | ||||||
|         ViewModel?.InitUpdateView(UpdateViewHandler); |         ViewModel?.InitUpdateView(UpdateViewHandler); | ||||||
| 
 | 
 | ||||||
|         txtRunningServerDisplay.Tapped += TxtRunningServerDisplay_Tapped; |         txtRunningServerDisplay.Tapped += TxtRunningServerDisplay_Tapped; | ||||||
|  |  | ||||||
|  | @ -59,6 +59,7 @@ public partial class AddGroupServerWindow | ||||||
| 
 | 
 | ||||||
|             this.BindCommand(ViewModel, vm => vm.SaveCmd, v => v.btnSave).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.SaveCmd, v => v.btnSave).DisposeWith(disposables); | ||||||
|         }); |         }); | ||||||
|  |         WindowsUtils.SetDarkBorder(this, AppManager.Instance.Config.UiItem.CurrentTheme); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async Task<bool> UpdateViewHandler(EViewAction action, object? obj) |     private async Task<bool> UpdateViewHandler(EViewAction action, object? obj) | ||||||
|  |  | ||||||
|  | @ -1,7 +1,6 @@ | ||||||
| using System.Reactive.Disposables; | using System.Reactive.Disposables; | ||||||
| using System.Windows.Input; | using System.Windows.Input; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using Splat; |  | ||||||
| 
 | 
 | ||||||
| namespace v2rayN.Views; | namespace v2rayN.Views; | ||||||
| 
 | 
 | ||||||
|  | @ -14,7 +13,6 @@ public partial class ClashProxiesView | ||||||
|     { |     { | ||||||
|         InitializeComponent(); |         InitializeComponent(); | ||||||
|         ViewModel = new ClashProxiesViewModel(UpdateViewHandler); |         ViewModel = new ClashProxiesViewModel(UpdateViewHandler); | ||||||
|         Locator.CurrentMutable.RegisterLazySingleton(() => ViewModel, typeof(ClashProxiesViewModel)); |  | ||||||
|         lstProxyDetails.PreviewMouseDoubleClick += lstProxyDetails_PreviewMouseDoubleClick; |         lstProxyDetails.PreviewMouseDoubleClick += lstProxyDetails_PreviewMouseDoubleClick; | ||||||
| 
 | 
 | ||||||
|         this.WhenActivated(disposables => |         this.WhenActivated(disposables => | ||||||
|  |  | ||||||
|  | @ -256,7 +256,7 @@ public partial class MainWindow | ||||||
|             case EGlobalHotkey.SystemProxySet: |             case EGlobalHotkey.SystemProxySet: | ||||||
|             case EGlobalHotkey.SystemProxyUnchanged: |             case EGlobalHotkey.SystemProxyUnchanged: | ||||||
|             case EGlobalHotkey.SystemProxyPac: |             case EGlobalHotkey.SystemProxyPac: | ||||||
|                 Locator.Current.GetService<StatusBarViewModel>()?.SetListenerType((ESysProxyType)((int)e - 1)); |                 AppEvents.SysProxyChangeRequested.OnNext((ESysProxyType)((int)e - 1)); | ||||||
|                 break; |                 break; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -294,11 +294,7 @@ public partial class MainWindow | ||||||
|                     var clipboardData = WindowsUtils.GetClipboardData(); |                     var clipboardData = WindowsUtils.GetClipboardData(); | ||||||
|                     if (clipboardData.IsNotEmpty()) |                     if (clipboardData.IsNotEmpty()) | ||||||
|                     { |                     { | ||||||
|                         var service = Locator.Current.GetService<MainWindowViewModel>(); |                         ViewModel?.AddServerViaClipboardAsync(clipboardData); | ||||||
|                         if (service != null) |  | ||||||
|                         { |  | ||||||
|                             _ = service.AddServerViaClipboardAsync(clipboardData); |  | ||||||
|                         } |  | ||||||
|                     } |                     } | ||||||
| 
 | 
 | ||||||
|                     break; |                     break; | ||||||
|  |  | ||||||
|  | @ -8,7 +8,6 @@ using System.Windows.Media; | ||||||
| using System.Windows.Threading; | using System.Windows.Threading; | ||||||
| using MaterialDesignThemes.Wpf; | using MaterialDesignThemes.Wpf; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using Splat; |  | ||||||
| using v2rayN.Base; | using v2rayN.Base; | ||||||
| using Point = System.Windows.Point; | using Point = System.Windows.Point; | ||||||
| 
 | 
 | ||||||
|  | @ -42,7 +41,6 @@ public partial class ProfilesView | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         ViewModel = new ProfilesViewModel(UpdateViewHandler); |         ViewModel = new ProfilesViewModel(UpdateViewHandler); | ||||||
|         Locator.CurrentMutable.RegisterLazySingleton(() => ViewModel, typeof(ProfilesViewModel)); |  | ||||||
| 
 | 
 | ||||||
|         this.WhenActivated(disposables => |         this.WhenActivated(disposables => | ||||||
|         { |         { | ||||||
|  |  | ||||||
|  | @ -3,7 +3,6 @@ using System.Windows; | ||||||
| using System.Windows.Input; | using System.Windows.Input; | ||||||
| using System.Windows.Threading; | using System.Windows.Threading; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using Splat; |  | ||||||
| using v2rayN.Manager; | using v2rayN.Manager; | ||||||
| 
 | 
 | ||||||
| namespace v2rayN.Views; | namespace v2rayN.Views; | ||||||
|  | @ -16,8 +15,8 @@ public partial class StatusBarView | ||||||
|     { |     { | ||||||
|         InitializeComponent(); |         InitializeComponent(); | ||||||
|         _config = AppManager.Instance.Config; |         _config = AppManager.Instance.Config; | ||||||
|         ViewModel = new StatusBarViewModel(UpdateViewHandler); |         ViewModel = StatusBarViewModel.Instance; | ||||||
|         Locator.CurrentMutable.RegisterLazySingleton(() => ViewModel, typeof(StatusBarViewModel)); |         ViewModel?.InitUpdateView(UpdateViewHandler); | ||||||
| 
 | 
 | ||||||
|         menuExit.Click += menuExit_Click; |         menuExit.Click += menuExit_Click; | ||||||
|         txtRunningServerDisplay.PreviewMouseDown += txtRunningInfoDisplay_MouseDoubleClick; |         txtRunningServerDisplay.PreviewMouseDown += txtRunningInfoDisplay_MouseDoubleClick; | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue