mirror of
				https://github.com/2dust/v2rayN.git
				synced 2025-10-26 18:24:43 +00:00 
			
		
		
		
	Compare commits
	
		
			5 commits
		
	
	
		
			71dcd8d1de
			...
			ab1dc45ed4
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | ab1dc45ed4 | ||
|   | 2d41272659 | ||
|   | e7f75010d3 | ||
|   | aa1ccdd01b | ||
|   | b17323c982 | 
					 9 changed files with 227 additions and 339 deletions
				
			
		|  | @ -1251,7 +1251,7 @@ public static class ConfigHandler | |||
|     public static async Task<ProfileItem?> GetPreSocksItem(Config config, ProfileItem node, ECoreType coreType) | ||||
|     { | ||||
|         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() | ||||
|             { | ||||
|  | @ -1262,7 +1262,7 @@ public static class ConfigHandler | |||
|                 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; | ||||
|             itemSocks = new ProfileItem() | ||||
|  |  | |||
|  | @ -32,19 +32,19 @@ public class ProfileItem : ReactiveObject | |||
|     public string GetSummary() | ||||
|     { | ||||
|         var summary = $"[{(ConfigType).ToString()}] "; | ||||
|         var arrAddr = Address.Contains(':') ? Address.Split(':') : Address.Split('.'); | ||||
|         var addr = arrAddr.Length switch | ||||
|         { | ||||
|             > 2 => $"{arrAddr.First()}***{arrAddr.Last()}", | ||||
|             > 1 => $"***{arrAddr.Last()}", | ||||
|             _ => Address | ||||
|         }; | ||||
|         if (ConfigType is EConfigType.Custom or > EConfigType.Group) | ||||
|         if (IsComplex()) | ||||
|         { | ||||
|             summary += $"[{CoreType.ToString()}]{Remarks}"; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             var arrAddr = Address.Contains(':') ? Address.Split(':') : Address.Split('.'); | ||||
|             var addr = arrAddr.Length switch | ||||
|             { | ||||
|                 > 2 => $"{arrAddr.First()}***{arrAddr.Last()}", | ||||
|                 > 1 => $"***{arrAddr.Last()}", | ||||
|                 _ => Address | ||||
|             }; | ||||
|             summary += $"{Remarks}({addr}:{Port})"; | ||||
|         } | ||||
|         return summary; | ||||
|  | @ -64,6 +64,51 @@ public class ProfileItem : ReactiveObject | |||
|         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 | ||||
| 
 | ||||
|     [PrimaryKey] | ||||
|  |  | |||
|  | @ -15,35 +15,8 @@ public partial class CoreConfigSingboxService(Config config) | |||
|         var ret = new RetResult(); | ||||
|         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 | ||||
|                 || node.Port <= 0) | ||||
|                 || !node.IsValid()) | ||||
|             { | ||||
|                 ret.Msg = ResUI.CheckServerSettings; | ||||
|                 return ret; | ||||
|  | @ -56,6 +29,17 @@ public partial class CoreConfigSingboxService(Config config) | |||
| 
 | ||||
|             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); | ||||
|             if (result.IsNullOrEmpty()) | ||||
|             { | ||||
|  | @ -169,12 +153,9 @@ public partial class CoreConfigSingboxService(Config config) | |||
|                     continue; | ||||
|                 } | ||||
|                 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 | ||||
|  | @ -214,27 +195,6 @@ public partial class CoreConfigSingboxService(Config config) | |||
|                 singboxConfig.inbounds.Add(inbound); | ||||
| 
 | ||||
|                 //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); | ||||
|                 if (server is null) | ||||
|                 { | ||||
|  | @ -293,7 +253,8 @@ public partial class CoreConfigSingboxService(Config config) | |||
|         var ret = new RetResult(); | ||||
|         try | ||||
|         { | ||||
|             if (node is not { Port: > 0 }) | ||||
|             if (node == null | ||||
|                 || !node.IsValid()) | ||||
|             { | ||||
|                 ret.Msg = ResUI.CheckServerSettings; | ||||
|                 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(); | ||||
|         try | ||||
|  | @ -405,53 +366,12 @@ public partial class CoreConfigSingboxService(Config config) | |||
|             await GenRouting(singboxConfig); | ||||
|             await GenExperimental(singboxConfig); | ||||
| 
 | ||||
|             var proxyProfiles = new List<ProfileItem>(); | ||||
|             foreach (var it in selecteds) | ||||
|             { | ||||
|                 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) | ||||
|             var groupRet = await GenGroupOutbound(parentNode, singboxConfig); | ||||
|             if (groupRet != 0) | ||||
|             { | ||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||
|                 return ret; | ||||
|             } | ||||
|             await GenOutboundsListWithChain(proxyProfiles, singboxConfig, multipleLoad); | ||||
| 
 | ||||
|             await GenDns(null, 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(); | ||||
|         try | ||||
|  | @ -503,48 +423,12 @@ public partial class CoreConfigSingboxService(Config config) | |||
|             await GenExperimental(singboxConfig); | ||||
|             singboxConfig.outbounds.RemoveAt(0); | ||||
| 
 | ||||
|             var proxyProfiles = new List<ProfileItem>(); | ||||
|             foreach (var it in selecteds) | ||||
|             { | ||||
|                 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) | ||||
|             var groupRet = await GenGroupOutbound(parentNode, singboxConfig); | ||||
|             if (groupRet != 0) | ||||
|             { | ||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||
|                 return ret; | ||||
|             } | ||||
|             await GenChainOutboundsList(proxyProfiles, singboxConfig); | ||||
| 
 | ||||
|             await GenDns(null, singboxConfig); | ||||
|             await ConvertGeo2Ruleset(singboxConfig); | ||||
|  |  | |||
|  | @ -204,6 +204,67 @@ public partial class CoreConfigSingboxService | |||
|         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) | ||||
|     { | ||||
|         try | ||||
|  |  | |||
|  | @ -385,29 +385,8 @@ public partial class CoreConfigSingboxService | |||
| 
 | ||||
|         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 ret = node.ConfigType switch | ||||
|             { | ||||
|                 EConfigType.PolicyGroup => | ||||
|                     await GenOutboundsListWithChain(childProfiles, singboxConfig, profileGroupItem.MultipleLoad, childBaseTagName), | ||||
|                 EConfigType.ProxyChain => | ||||
|                     await GenChainOutboundsList(childProfiles, singboxConfig, childBaseTagName), | ||||
|                 _ => throw new NotImplementedException() | ||||
|             }; | ||||
|             var ret = await GenGroupOutbound(node, singboxConfig, childBaseTagName); | ||||
|             if (ret == 0) | ||||
|             { | ||||
|                 return childBaseTagName; | ||||
|  |  | |||
|  | @ -15,35 +15,8 @@ public partial class CoreConfigV2rayService(Config config) | |||
|         var ret = new RetResult(); | ||||
|         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 | ||||
|                 || node.Port <= 0) | ||||
|                 || !node.IsValid()) | ||||
|             { | ||||
|                 ret.Msg = ResUI.CheckServerSettings; | ||||
|                 return ret; | ||||
|  | @ -57,6 +30,17 @@ public partial class CoreConfigV2rayService(Config config) | |||
| 
 | ||||
|             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); | ||||
|             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(); | ||||
| 
 | ||||
|  | @ -134,57 +118,12 @@ public partial class CoreConfigV2rayService(Config config) | |||
|             await GenDns(null, v2rayConfig); | ||||
|             await GenStatistic(v2rayConfig); | ||||
| 
 | ||||
|             var proxyProfiles = new List<ProfileItem>(); | ||||
|             foreach (var it in selecteds) | ||||
|             { | ||||
|                 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) | ||||
|             var groupRet = await GenGroupOutbound(parentNode, v2rayConfig); | ||||
|             if (groupRet != 0) | ||||
|             { | ||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||
|                 return ret; | ||||
|             } | ||||
|             await GenOutboundsListWithChain(proxyProfiles, v2rayConfig); | ||||
| 
 | ||||
|             //add balancers | ||||
|             await GenObservatory(v2rayConfig, multipleLoad); | ||||
|             await GenBalancer(v2rayConfig, multipleLoad); | ||||
| 
 | ||||
|             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(); | ||||
| 
 | ||||
|  | @ -284,48 +223,12 @@ public partial class CoreConfigV2rayService(Config config) | |||
|             await GenStatistic(v2rayConfig); | ||||
|             v2rayConfig.outbounds.RemoveAt(0); | ||||
| 
 | ||||
|             var proxyProfiles = new List<ProfileItem>(); | ||||
|             foreach (var it in selecteds) | ||||
|             { | ||||
|                 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) | ||||
|             var groupRet = await GenGroupOutbound(parentNode, v2rayConfig); | ||||
|             if (groupRet != 0) | ||||
|             { | ||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||
|                 return ret; | ||||
|             } | ||||
|             await GenChainOutboundsList(proxyProfiles, v2rayConfig); | ||||
| 
 | ||||
|             ret.Success = true; | ||||
| 
 | ||||
|  | @ -398,12 +301,9 @@ public partial class CoreConfigV2rayService(Config config) | |||
|                     continue; | ||||
|                 } | ||||
|                 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 | ||||
|  | @ -432,28 +332,6 @@ public partial class CoreConfigV2rayService(Config config) | |||
|                 it.Port = port; | ||||
|                 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 | ||||
|                 Inbounds4Ray inbound = new() | ||||
|                 { | ||||
|  | @ -464,6 +342,7 @@ public partial class CoreConfigV2rayService(Config config) | |||
|                 inbound.tag = inbound.protocol + inbound.port.ToString(); | ||||
|                 v2rayConfig.inbounds.Add(inbound); | ||||
| 
 | ||||
|                 //outbound | ||||
|                 var outbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); | ||||
|                 await GenOutbound(item, outbound); | ||||
|                 outbound.tag = Global.ProxyTag + inbound.port.ToString(); | ||||
|  | @ -497,7 +376,8 @@ public partial class CoreConfigV2rayService(Config config) | |||
|         var ret = new RetResult(); | ||||
|         try | ||||
|         { | ||||
|             if (node is not { Port: > 0 }) | ||||
|             if (node == null | ||||
|                 || !node.IsValid()) | ||||
|             { | ||||
|                 ret.Msg = ResUI.CheckServerSettings; | ||||
|                 return ret; | ||||
|  |  | |||
|  | @ -480,6 +480,69 @@ public partial class CoreConfigV2rayService | |||
|         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) | ||||
|     { | ||||
|         //fragment proxy | ||||
|  |  | |||
|  | @ -141,33 +141,8 @@ public partial class CoreConfigV2rayService | |||
| 
 | ||||
|         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 ret = node.ConfigType switch | ||||
|             { | ||||
|                 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); | ||||
|             } | ||||
|             var ret = await GenGroupOutbound(node, v2rayConfig, childBaseTagName); | ||||
|             if (ret == 0) | ||||
|             { | ||||
|                 return childBaseTagName; | ||||
|  |  | |||
|  | @ -59,6 +59,7 @@ public partial class AddGroupServerWindow | |||
| 
 | ||||
|             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) | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue