From 002364cbbd164373b5d755e0061353401024654a Mon Sep 17 00:00:00 2001 From: ihonliu Date: Tue, 12 Nov 2024 16:43:46 +0800 Subject: [PATCH 01/20] =?UTF-8?q?=E6=A3=80=E6=9F=A5=E6=96=B9=E6=B3=95?= =?UTF-8?q?=E6=98=AF=E5=90=A6=E4=B8=BA=E5=BC=82=E6=AD=A5=E6=96=B9=E6=B3=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../HttpServiceInvokerGenerator.cs | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs index 0af5869..29753b0 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs @@ -107,6 +107,18 @@ private static (MethodBuilder?, Diagnostic?) BuildMethod((IMethodSymbol, Attribu var authorize = (bool)(methodAttribute.GetNamedValue("Authorize") ?? false); needAuth = !allowsAnonymous && (authorize || controllerAuth); + // 检查当前返回类型是否是Task或Task + // 如果检查类型不符合要求,说明不是异步方法 + // 返回错误信息 + var returnTypeInfo = methodSymbol.ReturnType; + var isTask = returnTypeInfo.Name == "Task"; + var isGenericTask = returnTypeInfo.OriginalDefinition.ToDisplayString() == "System.Threading.Tasks.Task"; + + if (!isTask && !isGenericTask) + { + return (null, DiagnosticDefinitions.WAG00005(methodSymbol.Locations.FirstOrDefault())); + } + if (methodSymbol.HasAttribute(NotSupported)) { var b = MethodBuilder.Default From d16caf7c0916c63f55a855920adeaab6599fb7e4 Mon Sep 17 00:00:00 2001 From: ihonliu Date: Fri, 6 Dec 2024 16:52:05 +0800 Subject: [PATCH 02/20] =?UTF-8?q?feat:=20=E6=B7=BB=E5=8A=A0Analyzer?= =?UTF-8?q?=E7=9B=91=E6=B5=8B=E8=BF=BD=E8=B8=AA?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../AnalyzerReleases.Shipped.md | 3 + .../AnalyzerReleases.Unshipped.md | 15 ++ .../AutoWasmApi.Roslyn.csproj | 4 + .../DiagnosticDefinitions.cs | 233 +++++++++--------- 4 files changed, 139 insertions(+), 116 deletions(-) create mode 100644 src/AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md create mode 100644 src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md diff --git a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md new file mode 100644 index 0000000..d567f14 --- /dev/null +++ b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md @@ -0,0 +1,3 @@ +; Shipped analyzer releases +; https://github.com/dotnet/roslyn-analyzers/blob/master/src/Microsoft.CodeAnalysis.Analyzers/ReleaseTrackingAnalyzers.Help.md + diff --git a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md new file mode 100644 index 0000000..1884683 --- /dev/null +++ b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md @@ -0,0 +1,15 @@ +; Unshipped analyzer release +; https://github.com/dotnet/roslyn-analyzers/blob/master/src/Microsoft.CodeAnalysis.Analyzers/ReleaseTrackingAnalyzers.Help.md + +### New Rules +Rule ID | Category | Severity | Notes +--------|----------|----------|------- +WAG00001 | AutoWasmApiGenerator.ControllerGenerator | Error | 继承多个接口需要指定接口标注[WebControllerAttribute] +WAG00002 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 无法为该类型生成WebApi调用类,缺少接口 +WAG00003 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 方法参数过多 +WAG00004 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 控制器(controller)不能包含泛型 +WAG00005 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 仅支持异步方法 +WAG00006 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 路由中未包含路由参数({0}) +WAG00007 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 不能同时设置[FromBody]和[FromForm]({0}) +WAG00008 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 不能设置多个[FromBody]({0}) +WAG00009 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 暂不支持的返回值类型({0}) diff --git a/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj b/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj index 4ad31fb..7b2a2e3 100644 --- a/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj +++ b/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj @@ -9,5 +9,9 @@ + + + + diff --git a/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs b/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs index 780651b..993157c 100644 --- a/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs +++ b/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs @@ -7,121 +7,122 @@ namespace AutoWasmApiGenerator { internal class DiagnosticDefinitions { - /// - /// 继承多个接口需要指定接口标注[WebControllerAttribute] - /// - /// - /// - public static Diagnostic WAG00001(Location? location) => Diagnostic.Create(new DiagnosticDescriptor( - id: "WAG00001", - title: "继承多个接口需要指定接口标注[WebControllerAttribute]", - messageFormat: "继承多个接口需要指定接口标注[WebControllerAttribute]", - category: typeof(ControllerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Error, - isEnabledByDefault: true), location); - - /// - /// 无法为该类型生成WebApi调用类,缺少接口 - /// - /// - /// - public static Diagnostic WAG00002(Location? location) => Diagnostic.Create(new DiagnosticDescriptor( - id: "WAG00002", - title: "无法为该类型生成WebApi调用类,缺少接口", - messageFormat: "无法为该类型生成WebApi调用类,缺少接口", - category: typeof(HttpServiceInvokerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Error, - isEnabledByDefault: true), location); - - /// - /// 方法参数过多 - /// - /// - /// - public static Diagnostic WAG00003(Location? location) => Diagnostic.Create(new DiagnosticDescriptor( - id: "WAG00003", - title: "方法参数过多", - messageFormat: "方法参数过多", - category: typeof(HttpServiceInvokerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Error, - isEnabledByDefault: true), location); - - /// - /// 控制器(controller)不能包含泛型 - /// - /// - /// - public static Diagnostic WAG00004(Location? location) => Diagnostic.Create(new DiagnosticDescriptor( - id: "WAG00004", - title: "控制器(controller)不能包含泛型", - messageFormat: "控制器(controller)不能包含泛型", - category: typeof(HttpServiceInvokerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Error, - isEnabledByDefault: true), location); - - /// - /// 仅支持异步方法 - /// - /// - /// - public static Diagnostic WAG00005(Location? location) => Diagnostic.Create(new DiagnosticDescriptor( - id: "WAG00005", - title: "仅支持异步方法", - messageFormat: "仅支持异步方法", - category: typeof(HttpServiceInvokerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Error, - isEnabledByDefault: true), location); - - /// - /// 路由参数 - /// - /// - /// - public static Diagnostic WAG00006(Location? location, string? symbolString = null) => Diagnostic.Create(new DiagnosticDescriptor( - id: "WAG00006", - title: "路由中未包含路由参数", - messageFormat: $"路由中未包含路由参数({symbolString})", - category: typeof(HttpServiceInvokerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Error, - isEnabledByDefault: true), location); - - /// - /// 不能同时设置[FromBody]和[FromForm] - /// - /// - /// - public static Diagnostic WAG00007(Location? location, string? symbolString = null) => Diagnostic.Create(new DiagnosticDescriptor( - id: "WAG00007", - title: "不能同时设置[FromBody]和[FromForm]", - messageFormat: $"不能同时设置[FromBody]和[FromForm]({symbolString})", - category: typeof(HttpServiceInvokerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Error, - isEnabledByDefault: true), location); - - /// - /// 不能设置多个[FromBody] - /// - /// - /// - public static Diagnostic WAG00008(Location? location, string? symbolString = null) => Diagnostic.Create(new DiagnosticDescriptor( - id: "WAG00008", - title: "不能设置多个[FromBody]", - messageFormat: $"不能设置多个[FromBody]({symbolString})", - category: typeof(HttpServiceInvokerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Error, - isEnabledByDefault: true), location); - - /// - /// 暂不支持的返回值类型 - /// - /// - /// - public static Diagnostic WAG00009(Location? location, string? symbolString = null) => Diagnostic.Create(new DiagnosticDescriptor( - id: "WAG00009", - title: "暂不支持的返回值类型", - messageFormat: $"暂不支持的返回值类型({symbolString})", - category: typeof(HttpServiceInvokerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Error, - isEnabledByDefault: true), location); + private static readonly DiagnosticDescriptor Wag00001DiagDescriptor = new DiagnosticDescriptor( + id: "WAG00001", + title: "继承多个接口需要指定接口标注[WebControllerAttribute]", + messageFormat: "继承多个接口需要指定接口标注[WebControllerAttribute]", + category: typeof(ControllerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true); + + private static readonly DiagnosticDescriptor Wag00002DiagDescriptor = new DiagnosticDescriptor( + id: "WAG00002", + title: "无法为该类型生成WebApi调用类,缺少接口", + messageFormat: "无法为该类型生成WebApi调用类,缺少接口", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true); + + private static readonly DiagnosticDescriptor Wag00003DiagDescriptor = new DiagnosticDescriptor( + id: "WAG00003", + title: "方法参数过多", + messageFormat: "方法参数过多", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true); + + private static readonly DiagnosticDescriptor Wag00004DiagDescriptor = new DiagnosticDescriptor( + id: "WAG00004", + title: "控制器(controller)不能包含泛型", + messageFormat: "控制器(controller)不能包含泛型", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true); + + private static readonly DiagnosticDescriptor Wag00005DiagDescriptor = new DiagnosticDescriptor( + id: "WAG00005", + title: "仅支持异步方法", + messageFormat: "仅支持异步方法", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true); + + private static readonly DiagnosticDescriptor Wag00006DiagDescriptor = new DiagnosticDescriptor( + id: "WAG00006", + title: "路由中未包含路由参数", + messageFormat: "路由中未包含路由参数({0})", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true); + + private static readonly DiagnosticDescriptor Wag00007DiagDescriptor = new DiagnosticDescriptor( + id: "WAG00007", + title: "不能同时设置[FromBody]和[FromForm]", + messageFormat: "不能同时设置[FromBody]和[FromForm]({0})", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true); + + private static readonly DiagnosticDescriptor Wag00008DiagDescriptor = new DiagnosticDescriptor( + id: "WAG00008", + title: "不能设置多个[FromBody]", + messageFormat: "不能设置多个[FromBody]({0})", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true); + + + private static readonly DiagnosticDescriptor Wag00009DiagDescriptor = new DiagnosticDescriptor( + id: "WAG00009", + title: "暂不支持的返回值类型", + messageFormat: "暂不支持的返回值类型({0})", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true); + + public static Diagnostic WAG00001(Location? location) + { + return Diagnostic.Create(Wag00001DiagDescriptor, location); + } + + public static Diagnostic WAG00002(Location? location) + { + return Diagnostic.Create(Wag00002DiagDescriptor, location); + } + + public static Diagnostic WAG00003(Location? location) + { + return Diagnostic.Create(Wag00003DiagDescriptor, location); + } + + public static Diagnostic WAG00004(Location? location, string? symbolString = null) + { + return Diagnostic.Create(Wag00004DiagDescriptor, location, symbolString); + } + + public static Diagnostic WAG00005(Location? location, string? symbolString = null) + { + return Diagnostic.Create(Wag00005DiagDescriptor, location, symbolString); + } + + public static Diagnostic WAG00006(Location? location, string? symbolString = null) + { + return Diagnostic.Create(Wag00006DiagDescriptor, location, symbolString); + } + + public static Diagnostic WAG00007(Location? location, string? symbolString = null) + { + return Diagnostic.Create(Wag00007DiagDescriptor, location, symbolString); + } + + public static Diagnostic WAG00008(Location? location, string? symbolString = null) + { + return Diagnostic.Create(Wag00008DiagDescriptor, location, symbolString); + } + + public static Diagnostic WAG00009(Location? location, string? symbolString = null) + { + return Diagnostic.Create(Wag00009DiagDescriptor, location, symbolString); + } } } From a5630d0aa9c3cb733640ebf5d2e6cc93908e197b Mon Sep 17 00:00:00 2001 From: ihonliu Date: Fri, 6 Dec 2024 17:04:32 +0800 Subject: [PATCH 03/20] =?UTF-8?q?refactor:=20=E6=96=B0=E5=A2=9E=E9=94=99?= =?UTF-8?q?=E8=AF=AF=E4=BB=A3=E7=A0=81WAG00010=EF=BC=8C=E7=94=A8=E4=BA=8E?= =?UTF-8?q?=E6=98=BE=E7=A4=BA=E7=94=9F=E6=88=90=E5=99=A8=E7=94=9F=E6=88=90?= =?UTF-8?q?=E8=BF=87=E7=A8=8B=E4=B8=AD=E6=8A=9B=E5=87=BA=E7=9A=84=E5=BC=82?= =?UTF-8?q?=E5=B8=B8=E4=BF=A1=E6=81=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../AnalyzerReleases.Unshipped.md | 1 + src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs | 14 ++++++++++++++ .../HttpServiceInvokerGenerator.cs | 4 ++-- .../AutoWasmApiGenerator.csproj | 16 ++++++++-------- 4 files changed, 25 insertions(+), 10 deletions(-) diff --git a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md index 1884683..a0028c9 100644 --- a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md +++ b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md @@ -13,3 +13,4 @@ WAG00006 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 路由中 WAG00007 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 不能同时设置[FromBody]和[FromForm]({0}) WAG00008 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 不能设置多个[FromBody]({0}) WAG00009 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 暂不支持的返回值类型({0}) +WAG00010 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 生成服务调用器过程中发生错误: {0} diff --git a/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs b/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs index 993157c..cb6b642 100644 --- a/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs +++ b/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs @@ -80,6 +80,15 @@ internal class DiagnosticDefinitions defaultSeverity: DiagnosticSeverity.Error, isEnabledByDefault: true); + private static readonly DiagnosticDescriptor Wag00010DiagDescriptor = new( + id: "WAG00010", + title: "生成错误", + messageFormat: "生成过程中发生错误: {0}", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true + ); + public static Diagnostic WAG00001(Location? location) { return Diagnostic.Create(Wag00001DiagDescriptor, location); @@ -124,5 +133,10 @@ public static Diagnostic WAG00009(Location? location, string? symbolString = nul { return Diagnostic.Create(Wag00009DiagDescriptor, location, symbolString); } + + public static Diagnostic WAG00010(Location? location, string symbolString) + { + return Diagnostic.Create(Wag00010DiagDescriptor, location, symbolString); + } } } diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs index 29753b0..98172bc 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs @@ -165,9 +165,9 @@ private static (MethodBuilder?, Diagnostic?) BuildMethod((IMethodSymbol, Attribu useRouteParam = true; methodRoute = $"{methodScoped}/{customRoute}"; } - else + catch (Exception ex) { - methodRoute = customRoute!; + context.ReportDiagnostic(DiagnosticDefinitions.WAG00010(Location.None, ex.Message)); } //var methodRoute = $"{methodAttribute?.GetNamedValue("Route") ?? methodSymbol.Name.Replace("Async", "")}"; List statements = diff --git a/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj index ec5a6cc..7239835 100644 --- a/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj +++ b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj @@ -1,6 +1,6 @@  - + netstandard2.0;net6.0;net8.0 latest enable @@ -17,15 +17,15 @@ - + - + - + + + + + - - - - From 1016df9d9c6eec197388252137973624a4f1fd0e Mon Sep 17 00:00:00 2001 From: ihonliu Date: Fri, 6 Dec 2024 17:06:33 +0800 Subject: [PATCH 04/20] =?UTF-8?q?feat:=20=E4=B8=BA`IHttpClientHeaderHandle?= =?UTF-8?q?r`=E6=96=B0=E5=A2=9E`CancellationToken`=E5=8F=82=E6=95=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../IHttpClientHeaderHandler.cs | 18 ++++++++++-------- 1 file changed, 10 insertions(+), 8 deletions(-) diff --git a/src/AutoWasmApiGenerator/IHttpClientHeaderHandler.cs b/src/AutoWasmApiGenerator/IHttpClientHeaderHandler.cs index 2bb6003..83acf99 100644 --- a/src/AutoWasmApiGenerator/IHttpClientHeaderHandler.cs +++ b/src/AutoWasmApiGenerator/IHttpClientHeaderHandler.cs @@ -1,5 +1,6 @@ using System; using System.Net.Http; +using System.Threading; using System.Threading.Tasks; namespace AutoWasmApiGenerator; @@ -13,8 +14,9 @@ public interface IHttpClientHeaderHandler /// /// /// + /// /// - Task SetRequestHeaderAsync(HttpRequestMessage request); + Task SetRequestHeaderAsync(HttpRequestMessage request, CancellationToken token = default); } /// @@ -22,21 +24,21 @@ public interface IHttpClientHeaderHandler /// public class DefaultHttpClientHeaderHandler : IHttpClientHeaderHandler { - private DefaultHttpClientHeaderHandler() - { - - } - private static readonly Lazy lazy = new(() => new DefaultHttpClientHeaderHandler()); + private DefaultHttpClientHeaderHandler() { } + + private static readonly Lazy Lazy = new(() => new DefaultHttpClientHeaderHandler()); /// /// /// - public static IHttpClientHeaderHandler Default => lazy.Value; + public static IHttpClientHeaderHandler Default => Lazy.Value; + /// /// /// /// + /// /// - public Task SetRequestHeaderAsync(HttpRequestMessage request) + public Task SetRequestHeaderAsync(HttpRequestMessage request, CancellationToken token = default) { return Task.CompletedTask; } From 39b683ca5b0658d34ef3fdaa8b190b4bbb0ddc80 Mon Sep 17 00:00:00 2001 From: ihonliu Date: Fri, 6 Dec 2024 17:08:06 +0800 Subject: [PATCH 05/20] =?UTF-8?q?refactor:=20=E5=88=86=E7=A6=BBHttpService?= =?UTF-8?q?Invoker=E7=94=9F=E6=88=90=E9=80=BB=E8=BE=91=E8=87=B3=E7=8B=AC?= =?UTF-8?q?=E7=AB=8B=E7=B1=BB=E4=B8=AD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../HttpServiceInvokerGenerator.cs | 442 ++---------------- 1 file changed, 27 insertions(+), 415 deletions(-) diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs index 98172bc..9727610 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs @@ -1,441 +1,53 @@ -using Generators.Shared; -using Microsoft.CodeAnalysis; -using Microsoft.CodeAnalysis.CSharp.Syntax; -using System.Collections.Generic; +using System; using System.Linq; +using Generators.Shared; +using Microsoft.CodeAnalysis; using static AutoWasmApiGenerator.GeneratorHelpers; -using Generators.Shared.Builder; -using System.Diagnostics; -using System.Diagnostics.CodeAnalysis; -using System.Reflection.Metadata; -using System.Text.RegularExpressions; -namespace AutoWasmApiGenerator +namespace AutoWasmApiGenerator; + +[Generator(LanguageNames.CSharp)] +public class HttpServiceInvokerGenerator : IIncrementalGenerator { - [Generator(LanguageNames.CSharp)] - public class HttpServiceInvokerGenerator : IIncrementalGenerator + private static readonly HttpServiceInvokerGeneratorImpl Generator = new(); + + public void Initialize(IncrementalGeneratorInitializationContext context) { - public void Initialize(IncrementalGeneratorInitializationContext context) + context.RegisterSourceOutput(context.CompilationProvider, static (context, compilation) => { - //var list = context.SyntaxProvider.ForAttributeWithMetadataName( - // ApiInvokerAssemblyAttributeFullName, - // static (node, token) => true, - // static (c, t) => c); - - context.RegisterSourceOutput(context.CompilationProvider, static (context, compilation) => + try { - try + if (!compilation.Assembly.HasAttribute(ApiInvokerAssemblyAttributeFullName)) { - if (!compilation.Assembly.HasAttribute(ApiInvokerAssemblyAttributeFullName)) - { - return; - } - var all = compilation.GetAllSymbols(ApiInvokerAttributeFullName); - foreach (var item in all) - { - if (!item.HasAttribute(WebControllerAttributeFullName)) - { - continue; - } - if (CreateCodeFile(item, context, out var file)) - { -#if DEBUG - var ss = file.ToString(); -#endif - context.AddSource(file); - } - } - } - catch (System.Exception ex) - { - context.ReportDiagnostic(Diagnostic.Create(new DiagnosticDescriptor( - id: "ERROR00001", - title: "生成错误", - messageFormat: ex.Message, - category: typeof(HttpServiceInvokerGenerator).FullName!, - defaultSeverity: DiagnosticSeverity.Warning, - isEnabledByDefault: true - ), Location.None)); + return; } - }); - } - private static bool CreateCodeFile(INamedTypeSymbol interfaceSymbol, SourceProductionContext context, [NotNullWhen(true)] out CodeFile? file) - { - var methods = interfaceSymbol.GetAllMethodWithAttribute(WebMethodAttributeFullName); - List members = new List(); - _ = interfaceSymbol.GetAttribute(WebControllerAttributeFullName, out var controllerAttrData); - var ns = NamespaceBuilder.Default.Namespace(interfaceSymbol.ContainingNamespace.ToDisplayString()); - var invokeClass = CreateHttpClassBuilder(interfaceSymbol); - var scopeName = interfaceSymbol.FormatClassName(); - var route = controllerAttrData.GetNamedValue("Route") as string; - bool needAuth = (bool)(controllerAttrData.GetNamedValue("Authorize") ?? false); - foreach (var method in methods) - { - var result = BuildMethod(method, route, scopeName, needAuth, out var n); - if (n && !needAuth) - { - needAuth = true; - } - if (result.Item2 != null) + var all = compilation.GetAllSymbols(ApiInvokerAttributeFullName); + foreach (var item in all) { - context.ReportDiagnostic(result.Item2); - file = null; - return false; - } - members.Add(result.Item1!); - } - - var fields = BuildField(needAuth); - var constructor = BuildConstructor(interfaceSymbol, needAuth); - members.AddRange(fields); - members.Add(constructor); - - file = CodeFile.New($"{interfaceSymbol.FormatFileName()}ApiInvoker.g.cs") - .AddUsings("using Microsoft.Extensions.DependencyInjection;") - .AddMembers(ns.AddMembers(invokeClass.AddMembers([.. members]))); - - return true; - } - - private static (MethodBuilder?, Diagnostic?) BuildMethod((IMethodSymbol, AttributeData?) method, string? route, string scopeName, bool controllerAuth, out bool needAuth) - { - var methodSymbol = method.Item1; - var methodAttribute = method.Item2; - - var allowsAnonymous = (bool)(methodAttribute.GetNamedValue("AllowAnonymous") ?? false); - var authorize = (bool)(methodAttribute.GetNamedValue("Authorize") ?? false); - needAuth = !allowsAnonymous && (authorize || controllerAuth); - - // 检查当前返回类型是否是Task或Task - // 如果检查类型不符合要求,说明不是异步方法 - // 返回错误信息 - var returnTypeInfo = methodSymbol.ReturnType; - var isTask = returnTypeInfo.Name == "Task"; - var isGenericTask = returnTypeInfo.OriginalDefinition.ToDisplayString() == "System.Threading.Tasks.Task"; - - if (!isTask && !isGenericTask) - { - return (null, DiagnosticDefinitions.WAG00005(methodSymbol.Locations.FirstOrDefault())); - } - - if (methodSymbol.HasAttribute(NotSupported)) - { - var b = MethodBuilder.Default - .MethodName(methodSymbol.Name) - .Generic([.. methodSymbol.GetTypeParameters()]) - .ReturnType(methodSymbol.ReturnType.ToDisplayString()) - .AddParameter([.. methodSymbol.Parameters.Select(p => $"{p.Type.ToDisplayString()} {p.Name}")]) - .AddGeneratedCodeAttribute(typeof(HttpServiceInvokerGenerator)) - .Lambda("throw new global::System.NotSupportedException()"); - return (b, null); - } - - // 检查当前返回类型是否是Task或Task - // 如果检查类型不符合要求,说明不是异步方法 - // 返回错误信息 - var returnTypeInfo = methodSymbol.ReturnType; - var isTask = returnTypeInfo.Name == "Task"; - var isGenericTask = returnTypeInfo.OriginalDefinition.ToDisplayString() == "System.Threading.Tasks.Task"; - - if (!isTask && !isGenericTask) - { - return (null, DiagnosticDefinitions.WAG00005(methodSymbol.Locations.FirstOrDefault())); - } - - string webMethod; - if (!methodAttribute.GetNamedValue("Method", out var v)) - { - webMethod = "Post"; - } - else - { - webMethod = WebMethod[(int)v!]; - } - var methodScoped = methodSymbol.Name.Replace("Async", ""); - var customRoute = methodAttribute?.GetNamedValue("Route")?.ToString(); - string methodRoute; - bool useRouteParam = false; - if (string.IsNullOrEmpty(customRoute)) - { - methodRoute = methodScoped; - } - else if (Regex.Match(customRoute, "{.+}").Success) - { - useRouteParam = true; - methodRoute = $"{methodScoped}/{customRoute}"; - } - catch (Exception ex) - { - context.ReportDiagnostic(DiagnosticDefinitions.WAG00010(Location.None, ex.Message)); - } - //var methodRoute = $"{methodAttribute?.GetNamedValue("Route") ?? methodSymbol.Name.Replace("Async", "")}"; - List statements = - [ - // var url = ""; - // var client = clientFactory.CreateClient(nameof()); - $"var _client_gen = this.clientFactory.CreateClient(\"{scopeName}\")", - // var request = new HttpRequestMessage(); - "var _request_gen = new global::System.Net.Http.HttpRequestMessage()", - // request.Method = HttpMethod. - $"_request_gen.Method = global::System.Net.Http.HttpMethod.{webMethod}", - ]; - if (needAuth) - { - statements.Add("await headerHandler.SetRequestHeaderAsync(_request_gen)"); - } - - // 处理参数标签 - var paramInfos = methodSymbol.Parameters.Select(p => - { - if (p.GetAttribute(WebMethodParameterBindingAttribute, out var ad)) - { - ad!.GetConstructorValue(0, out var bindingType); - var t = (int)bindingType!; - return ((int)bindingType!, p); - } - if (useRouteParam && customRoute!.Contains(p.Name)) - { - return (1, p); - } - return (-1, p); - }); - - // 0 - Query - // 1 - Route - // 2 - Form - // 3 - Body - // 4 - Header - - #region 检查参数配置 - - var routerParameters = paramInfos.Where(t => t.Item1 == 1); - foreach ((int _, IParameterSymbol p) item in routerParameters) - { - if (!methodRoute.Contains($"{{{item.p.Name}}}")) - { - return (null, DiagnosticDefinitions.WAG00006(methodSymbol.Locations.FirstOrDefault(), methodSymbol.ToDisplayString())); - } - } - - if (paramInfos.Any(t => t.Item1 == 2) && paramInfos.Any(t => t.Item1 == 3)) - { - return (null, DiagnosticDefinitions.WAG00007(methodSymbol.Locations.FirstOrDefault(), methodSymbol.ToDisplayString())); - } - #endregion - - var url = $""" - var _url_gen = $"api/{route ?? scopeName}/{methodRoute}" - """; - statements.Add(url); - - #region 处理Query参数 - - var queryParameters = paramInfos.Where(t => (t.Item1 == -1 && webMethod == "Get") || t.Item1 == 0); - - if (queryParameters.Any()) - { - statements.Add("var _queries_gen = new List()"); - foreach (var item in queryParameters) - { - var p = item.p; - if (p.Type is INamedTypeSymbol { TypeKind: TypeKind.Class, SpecialType: not SpecialType.System_String } parameterClassType) - { - var properties = parameterClassType.GetMembers().Where(m => m.Kind == SymbolKind.Property); - foreach (var prop in properties) - { - statements.Add($$"""_queries_gen.Add($"{nameof({{p.Name}}.{{prop.Name}})}={{{p.Name}}.{{prop.Name}}}")"""); - } - } - else - { - statements.Add($$"""_queries_gen.Add($"{nameof({{p.Name}})}={{{p.Name}}}")"""); - } - } - var setUrl = """ - _url_gen = $"{_url_gen}?{string.Join("&", _queries_gen)}" - """; - statements.Add(setUrl); - } - - #endregion - - #region 处理Form参数 - - var formParameters = paramInfos.Where(t => t.Item1 == 2); - - if (formParameters.Any()) - { - statements.Add("var _formDatas_gen = new List>()"); - foreach (var item in formParameters) - { - var p = item.p; - if (p.Type is INamedTypeSymbol { TypeKind: TypeKind.Class, SpecialType: not SpecialType.System_String } parameterClassType) - { - var properties = parameterClassType.GetMembers().Where(m => m.Kind == SymbolKind.Property); - foreach (var prop in properties) - { - statements.Add($$""" - _formDatas_gen.Add(new global::System.Collections.Generic.KeyValuePair(nameof({{p.Name}}.{{prop.Name}}), $"{{{p.Name}}.{{prop.Name}}}")) - """); - } - } - else + if (!item.HasAttribute(WebControllerAttributeFullName)) { - statements.Add($$"""_formDatas_gen.Add(new global::System.Collections.Generic.KeyValuePair(nameof({{p.Name}}), $"{{{p.Name}}}"))"""); + continue; } - } - statements.Add("var _formContent_gen = new global::System.Net.Http.FormUrlEncodedContent(_formDatas_gen)"); - statements.Add("_formContent_gen.Headers.ContentType = new(\"application/x-www-form-urlencoded\")"); - statements.Add("""_request_gen.Content = _formContent_gen"""); - - } - - #endregion - - #region 处理Body参数 - - var bodyParameters = paramInfos.Where(t => (t.Item1 == -1 && webMethod != "Get") || t.Item1 == 3); - if (bodyParameters.Count() > 1) - { - return (null, DiagnosticDefinitions.WAG00008(methodSymbol.Locations.FirstOrDefault(), methodSymbol.ToDisplayString())); - } - - if (bodyParameters.Any()) - { - var p = bodyParameters.First().p; - statements.Add($"var _json_gen = global::System.Text.Json.JsonSerializer.Serialize({p.Name})"); - statements.Add("""_request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json")"""); - } - - #endregion - - #region 处理Header参数 - - var headerParameters = paramInfos.Where(t => t.Item1 == 4); - if (headerParameters.Any()) - { - foreach (var item in headerParameters) - { - var p = item.p; - if (p.Type is INamedTypeSymbol { TypeKind: TypeKind.Class, SpecialType: not SpecialType.System_String } parameterClassType) + var (generatedFileName, sourceCode, errorAndWarnings) = Generator.Generate(item); + if (errorAndWarnings.Any()) { - var properties = parameterClassType.GetMembers().Where(m => m.Kind == SymbolKind.Property); - foreach (var prop in properties) + foreach (var item1 in errorAndWarnings) { - statements.Add($$"""_request_gen.Headers.Add(nameof({{p.Name}}.{{prop.Name}}), $"{{{p.Name}}.{{prop.Name}}}")"""); + context.ReportDiagnostic(item1); } } else { - statements.Add($$"""_request_gen.Headers.Add(nameof({{p.Name}}), $"{{{p.Name}}}")"""); - } - } - } - - #endregion - - statements.Add("_request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative)"); - var returnType = methodSymbol.ReturnType.GetGenericTypes().FirstOrDefault() ?? methodSymbol.ReturnType; - - if (methodSymbol.ReturnsVoid || returnType.ToDisplayString() == "System.Threading.Tasks.Task") - { - statements.Add("_ = await _client_gen.SendAsync(_request_gen)"); - } - else - { - statements.Add("var _response_gen = await _client_gen.SendAsync(_request_gen)"); - statements.Add("_response_gen.EnsureSuccessStatusCode()"); - // 返回值是复杂类型,使用Json反序列化 - if (returnType is { TypeKind: TypeKind.Class, SpecialType: not SpecialType.System_String }) - { - statements.Add("var _stream_gen = await _response_gen.Content.ReadAsStreamAsync()"); - //return System.Text.Json.JsonSerializer.Deserialize(jsonStream, jsonOptions); - statements.Add($"return global::System.Text.Json.JsonSerializer.Deserialize<{returnType.ToDisplayString()}>(_stream_gen, _JSON_OPTIONS_gen);"); - } - else - { - statements.Add("var _str_gen = await _response_gen.Content.ReadAsStringAsync()"); - if (returnType.SpecialType == SpecialType.System_String) - { - statements.Add("return _str_gen"); - } - else if (returnType.IsValueType) - { - statements.Add($"{returnType.ToDisplayString()}.TryParse(_str_gen, out var val)"); - statements.Add("return val"); - } - else - { - return (null, DiagnosticDefinitions.WAG00009(methodSymbol.Locations.FirstOrDefault())); + context.AddSource(generatedFileName, sourceCode); } } } - - var builder = MethodBuilder.Default - .MethodName(methodSymbol.Name) - .Generic([.. methodSymbol.GetTypeParameters()]) - .Async() - .ReturnType(methodSymbol.ReturnType.ToDisplayString()) - .AddParameter([.. methodSymbol.Parameters.Select(p => $"{p.Type.ToDisplayString()} {p.Name}")]) - .AddGeneratedCodeAttribute(typeof(HttpServiceInvokerGenerator)) - .AddBody([.. statements]); - return (builder, null); - } - - private static IEnumerable BuildField(bool needAuth) - { - // private readonly JsonSerializerOptions jsonOptions; - yield return FieldBuilder.Default.MemberType("global::System.Text.Json.JsonSerializerOptions") - .FieldName("_JSON_OPTIONS_gen"); - // private readonly IHttpClientFactory clientFactory; - yield return FieldBuilder.Default - .MemberType("global::System.Net.Http.IHttpClientFactory") - .FieldName("clientFactory"); - if (needAuth) - { - yield return FieldBuilder.Default - .MemberType("global::AutoWasmApiGenerator.IHttpClientHeaderHandler") - .FieldName("headerHandler"); - } - } - - private static ConstructorBuilder BuildConstructor(INamedTypeSymbol classSymbol, bool needAuth) - { - List parameters = [ - "global::System.Net.Http.IHttpClientFactory factory" - ]; - List body = ["clientFactory = factory;"]; - if (needAuth) - { - //parameters.Add("global::AutoWasmApiGenerator.IHttpClientHeaderHandler handler"); - parameters.Add("global::System.IServiceProvider services"); - body.Add("headerHandler = services.GetService() ?? global::AutoWasmApiGenerator.DefaultHttpClientHeaderHandler.Default"); - } - - return ConstructorBuilder.Default - .MethodName($"{FormatClassName(classSymbol.MetadataName)}ApiInvoker") - .AddParameter([.. parameters]) - .AddBody([.. body]) - .AddBody("_JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true };"); - } - - private static ClassBuilder CreateHttpClassBuilder(INamedTypeSymbol interfaceSymbol) - { - IEnumerable additionalAttribute = []; - if (interfaceSymbol.GetAttribute(ApiInvokerAttributeFullName, out var data)) + catch (Exception ex) { - //var o = data.GetAttributeValue(nameof(ApiInvokerGeneraAttribute.Attribute)); - additionalAttribute = interfaceSymbol.GetAttributeInitInfo(ApiInvokerAttributeFullName, data!); + context.ReportDiagnostic(DiagnosticDefinitions.WAG00010(Location.None, ex.Message)); } - - - return ClassBuilder.Default - .ClassName($"{FormatClassName(interfaceSymbol.MetadataName)}ApiInvoker") - .AddGeneratedCodeAttribute(typeof(HttpServiceInvokerGenerator)) - .Attribute([.. additionalAttribute.Select(i => i.ToString())]) - .BaseType(interfaceSymbol.ToDisplayString()); - } + }); } -} +} \ No newline at end of file From c024a40c9ab47e88c56fb74b167508965b4f9cb5 Mon Sep 17 00:00:00 2001 From: ihonliu Date: Fri, 6 Dec 2024 17:11:09 +0800 Subject: [PATCH 06/20] =?UTF-8?q?bug:=20=E4=BD=BF=E7=94=A8global::?= =?UTF-8?q?=E5=91=BD=E5=90=8D=E5=9F=9F=E9=81=BF=E5=85=8D=E7=94=9F=E6=88=90?= =?UTF-8?q?=E7=9A=84=E4=BB=A3=E7=A0=81=E5=9B=A0=E5=91=BD=E5=90=8D=E5=9F=9F?= =?UTF-8?q?=E5=AF=BC=E8=87=B4=E7=9A=84=E7=BC=96=E8=AF=91=E9=94=99=E8=AF=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../HttpServiceInvokerGeneratorImpl.cs | 477 ++++++++++++++++++ .../IHttpServiceInvokerGenerator.cs | 10 + 2 files changed, 487 insertions(+) create mode 100644 src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs create mode 100644 src/AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs new file mode 100644 index 0000000..6cd7095 --- /dev/null +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs @@ -0,0 +1,477 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Diagnostics.CodeAnalysis; +using System.Linq; +using System.Text.RegularExpressions; +using System.Threading; +using Generators.Models; +using Generators.Shared; +using Generators.Shared.Builder; +using Microsoft.CodeAnalysis; +using static AutoWasmApiGenerator.GeneratorHelpers; + +namespace AutoWasmApiGenerator; + +public class HttpServiceInvokerGeneratorImpl : IHttpServiceInvokerGenerator +{ + // 0 - Query + // 1 - Route + // 2 - Form + // 3 - Body + // 4 - Header + private static class WebMethodConstants + { + public const string Get = "Get"; + public const string Post = "Post"; + public const string Put = "Put"; + public const string Delete = "Delete"; + + public const int Ignore = -1; + public const int FromQuery = 0; + public const int FromRoute = 1; + public const int FromForm = 2; + public const int FromBody = 3; + public const int FromHeader = 4; + } + + public (string generatedFileName, string sourceCode, List errorAndWarnings) Generate( + INamedTypeSymbol interfaceSymbol) + { + var errorAndWarnings = new List(); + var ret = CreateCodeFile(interfaceSymbol, errorAndWarnings, out var file); + return ret + ? (file!.FileName, file.ToString(), errorAndWarnings) + : ("", "", errorAndWarnings); + } + + private static bool CreateCodeFile(INamedTypeSymbol interfaceSymbol, List errorAndWarnings, + [NotNullWhen(true)] out CodeFile? file) + { + var methods = interfaceSymbol.GetAllMethodWithAttribute(WebMethodAttributeFullName); + List members = new(); + _ = interfaceSymbol.GetAttribute(WebControllerAttributeFullName, out var controllerAttrData); + var invokeClass = CreateHttpClassBuilder(interfaceSymbol); + var scopeName = interfaceSymbol.FormatClassName(); + var route = controllerAttrData.GetNamedValue("Route") as string; + var needAuth = (bool)(controllerAttrData.GetNamedValue("Authorize") ?? false); + foreach (var method in methods) + { + var methodBuilder = BuildMethod(method, route, scopeName, needAuth, out var n, errorAndWarnings); + if (n && !needAuth) + { + needAuth = true; + } + + if (errorAndWarnings.Count > 0) + { + // Debug.Assert(methodBuilder == null); + file = null; + return false; + } + + members.Add(methodBuilder!); + } + + var fields = BuildField(needAuth); + var constructor = BuildConstructor(interfaceSymbol, needAuth); + members.AddRange(fields); + members.Add(constructor); + + var classBuilder = invokeClass.AddMembers([.. members]); + var ns = NamespaceBuilder.Default.Namespace(interfaceSymbol.ContainingNamespace.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)); + var namespaceBuilder = ns.AddMembers(classBuilder); + file = CodeFile.New($"{interfaceSymbol.FormatFileName()}ApiInvoker.g.cs") + .AddUsings("using Microsoft.Extensions.DependencyInjection;") + .AddFileHeader( +"// ", +"#pragma warning disable", +"#nullable enable") + .AddMembers(namespaceBuilder); + + return true; + } + + private static MethodBuilder? BuildMethod((IMethodSymbol, AttributeData?) method, string? route, + string scopeName, bool controllerAuth, out bool needAuth, List errorAndWarnings) + { + var methodSymbol = method.Item1; + var methodAttribute = method.Item2; + + var allowsAnonymous = (bool)(methodAttribute.GetNamedValue("AllowAnonymous") ?? false); + var authorize = (bool)(methodAttribute.GetNamedValue("Authorize") ?? false); + needAuth = !allowsAnonymous && (authorize || controllerAuth); + var cancellationTokenName = methodSymbol.Parameters.Where(p => p.Type.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat) == "global::System.Threading.CancellationToken").Select(p => p.Name).FirstOrDefault(); + var hasCancellationToken = cancellationTokenName != null; + + if (methodSymbol.HasAttribute(NotSupported)) + { + TypeParameterInfo[] typeParameterInfos = [.. methodSymbol.GetTypeParameters()]; + var rt = methodSymbol.ReturnType; + var type = rt.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat); + string[] parameters = [.. methodSymbol.Parameters.Select(p => $"{p.Type.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)} {p.Name}")]; + var b = MethodBuilder.Default + .MethodName(methodSymbol.Name) + .Generic(typeParameterInfos) + .ReturnType(type) + .AddParameter(parameters) + .AddGeneratedCodeAttribute(typeof(HttpServiceInvokerGenerator)) + .Lambda("throw new global::System.NotSupportedException()"); + return b; + } + + // 检查当前返回类型是否是Task或Task + // 如果检查类型不符合要求,说明不是异步方法 + // 返回错误信息 + var returnTypeInfo = methodSymbol.ReturnType; + var isTask = returnTypeInfo.Name == "Task"; + var isGenericTask = returnTypeInfo.OriginalDefinition.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat) + .StartsWith("global::System.Threading.Tasks.Task<"); + + if (!isTask && !isGenericTask) + { + errorAndWarnings.Add(DiagnosticDefinitions.WAG00005(methodSymbol.Locations.FirstOrDefault())); + return null; + } + + var webMethod = methodAttribute.GetNamedValue("Method", out var v) ? WebMethod[(int)v!] : "Post"; + var methodScoped = methodSymbol.Name.Replace("Async", ""); + var customRoute = methodAttribute?.GetNamedValue("Route")?.ToString(); + string methodRoute; + var useRouteParam = false; + if (string.IsNullOrEmpty(customRoute)) + { + methodRoute = methodScoped; + } + else if (Regex.Match(customRoute, "{.+}").Success) + { + useRouteParam = true; + methodRoute = $"{methodScoped}/{customRoute}"; + } + else + { + methodRoute = customRoute!; + } + + //var methodRoute = $"{methodAttribute?.GetNamedValue("Route") ?? methodSymbol.Name.Replace("Async", "")}"; + List statements = + [ + // var url = ""; + // var client = clientFactory.CreateClient(nameof()); + $"var _client_gen = this.clientFactory.CreateClient(\"{scopeName}\")", + // var request = new HttpRequestMessage(); + "var _request_gen = new global::System.Net.Http.HttpRequestMessage()", + // request.Method = HttpMethod. + $"_request_gen.Method = global::System.Net.Http.HttpMethod.{webMethod}" + ]; + if (needAuth) + { + statements.Add($"await headerHandler.SetRequestHeaderAsync(_request_gen, {cancellationTokenName ?? "global::System.Threading.CancellationToken.None"})"); + } + + // 处理参数标签 + var paramInfos = methodSymbol.Parameters.Select(p => + { + if (p.GetAttribute(WebMethodParameterBindingAttribute, out var ad)) + { + ad!.GetConstructorValue(0, out var bindingType); + var t = (int)bindingType!; + return (t, p); + } + + if (useRouteParam && customRoute!.Contains(p.Name)) + { + return (1, p); + } + + return (-1, p); + }).ToList(); + + #region 检查参数配置 + + var routerParameters = paramInfos.Where(t => t.Item1 == WebMethodConstants.FromRoute); + foreach ((int _, IParameterSymbol p) item in routerParameters) + { + if (methodRoute.Contains($"{{{item.p.Name}}}")) continue; + errorAndWarnings.Add(DiagnosticDefinitions.WAG00006(methodSymbol.Locations.FirstOrDefault(), + methodSymbol.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat))); + return null; + } + + if (paramInfos.Any(t => t.Item1 == 2) && paramInfos.Any(t => t.Item1 == 3)) + { + errorAndWarnings.Add(DiagnosticDefinitions.WAG00006(methodSymbol.Locations.FirstOrDefault(), + methodSymbol.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat))); + return null; + } + + #endregion + + var url = $""" + var _url_gen = $"api/{route ?? scopeName}/{methodRoute}" + """; + statements.Add(url); + + AddQueryParameters(statements, paramInfos, webMethod, errorAndWarnings); + AddFormParameters(statements, paramInfos); + AddBodyParameters(statements, paramInfos, webMethod, methodSymbol, errorAndWarnings); + AddHeaderParameters(statements, paramInfos); + + statements.Add("_request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative)"); + var returnType = methodSymbol.ReturnType.GetGenericTypes().FirstOrDefault() ?? methodSymbol.ReturnType; + + if (methodSymbol.ReturnsVoid || (isTask && !isGenericTask)) + { + statements.Add(hasCancellationToken + ? $"_ = await _client_gen.SendAsync(_request_gen, {cancellationTokenName})" + : "_ = await _client_gen.SendAsync(_request_gen)"); + } + else + { + statements.Add(hasCancellationToken + ? $"var _response_gen = await _client_gen.SendAsync(_request_gen, {cancellationTokenName})" + : "var _response_gen = await _client_gen.SendAsync(_request_gen)"); + statements.Add("_response_gen.EnsureSuccessStatusCode()"); + AddResponseHandling(statements, returnType, errorAndWarnings, cancellationTokenName); + } + + var parameter = methodSymbol.Parameters.Select(p => $"{p.Type.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)} {p.Name}").ToArray(); + var builder = MethodBuilder.Default + .MethodName(methodSymbol.Name) + .Generic(methodSymbol.GetTypeParameters().ToArray()) + .Async() + .ReturnType(methodSymbol.ReturnType.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)) + .AddParameter(parameter) + .AddGeneratedCodeAttribute(typeof(HttpServiceInvokerGenerator)) + .AddBody(statements.ToArray()); + return builder; + } + + private static IEnumerable BuildField(bool needAuth) + { + // private readonly JsonSerializerOptions jsonOptions; + yield return FieldBuilder.Default.MemberType("global::System.Text.Json.JsonSerializerOptions") + .FieldName("_JSON_OPTIONS_gen"); + // private readonly IHttpClientFactory clientFactory; + yield return FieldBuilder.Default + .MemberType("global::System.Net.Http.IHttpClientFactory") + .FieldName("clientFactory"); + if (needAuth) + { + yield return FieldBuilder.Default + .MemberType("global::AutoWasmApiGenerator.IHttpClientHeaderHandler") + .FieldName("headerHandler"); + } + } + + /// + /// 处理Query参数 + /// + /// + /// + /// + /// + private static void AddQueryParameters(List statements, + IEnumerable<(int i, IParameterSymbol p)> paramInfos, string webMethod, List errorAndWarnings) + { + var queryParameters = paramInfos.Where(t => (t.i == -1 && webMethod == WebMethodConstants.Get) || t.i == WebMethodConstants.FromQuery).ToList(); + + if (!queryParameters.Any()) return; + statements.Add("var _queries_gen = new global::System.Collections.Generic.List()"); + foreach (var item in queryParameters) + { + var p = item.p; + if (p.Type is INamedTypeSymbol + { + TypeKind: TypeKind.Class, SpecialType: not SpecialType.System_String + } parameterClassType) + { + var properties = parameterClassType.GetMembers().Where(m => m.Kind == SymbolKind.Property); + foreach (var prop in properties) + { + statements.Add( + $$"""_queries_gen.Add($"{nameof({{p.Name}}.{{prop.Name}})}={{{p.Name}}.{{prop.Name}}}")"""); + } + } + else + { + statements.Add($$"""_queries_gen.Add($"{nameof({{p.Name}})}={{{p.Name}}}")"""); + } + } + + var setUrl = """ + _url_gen = $"{_url_gen}?{string.Join("&", _queries_gen)}" + """; + statements.Add(setUrl); + } + + /// + /// 处理Form参数 + /// + /// + /// + private static void AddFormParameters(List statements, IEnumerable<(int i, IParameterSymbol p)> paramInfos) + { + var formParameters = paramInfos.Where(t => t.i == WebMethodConstants.FromForm).ToList(); + + if (!formParameters.Any()) return; + statements.Add( + "var _formData_gen = new List>()"); + foreach (var item in formParameters) + { + var p = item.p; + if (p.Type is INamedTypeSymbol + { + TypeKind: TypeKind.Class, SpecialType: not SpecialType.System_String + } parameterClassType) + { + var properties = parameterClassType.GetMembers().Where(m => m.Kind == SymbolKind.Property); + foreach (var prop in properties) + { + statements.Add($$""" + _formData_gen.Add(new global::System.Collections.Generic.KeyValuePair(nameof({{p.Name}}.{{prop.Name}}), $"{{{p.Name}}.{{prop.Name}}}")) + """); + } + } + else + { + statements.Add( + $$"""_formData_gen.Add(new global::System.Collections.Generic.KeyValuePair(nameof({{p.Name}}), $"{{{p.Name}}}"))"""); + } + } + + statements.Add("var _formContent_gen = new global::System.Net.Http.FormUrlEncodedContent(_formData_gen)"); + statements.Add("_formContent_gen.Headers.ContentType = new(\"application/x-www-form-urlencoded\")"); + statements.Add("_request_gen.Content = _formContent_gen"); + } + + /// + /// 处理Body参数 + /// + /// + /// + /// + /// + /// + /// + private static bool AddBodyParameters(List statements, IEnumerable<(int i, IParameterSymbol p)> paramInfos, string webMethod, IMethodSymbol methodSymbol, List errorAndWarnings) + { + // var bodyParameters = paramInfos.Where(t => (t.i == -1 && webMethod != WebMethodConstants.Get) || t.i == WebMethodConstants.Body).ToList(); + var bodyParameters = paramInfos.Where(t => t.i == WebMethodConstants.FromBody).ToList(); + if (bodyParameters.Count > 1) + { + errorAndWarnings.Add(DiagnosticDefinitions.WAG00008(methodSymbol.Locations.FirstOrDefault())); + return false; + } + + if (bodyParameters.Count == 0) return true; + var p = bodyParameters.First().p; + statements.Add($"var _json_gen = global::System.Text.Json.JsonSerializer.Serialize({p.Name})"); + statements.Add( + """_request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json")"""); + return true; + } + + /// + /// 处理Header参数 + /// + /// + /// + private static bool AddHeaderParameters(List statements, IEnumerable<(int i, IParameterSymbol p)> paramInfos) + { + var headerParameters = paramInfos.Where(t => t.Item1 == WebMethodConstants.FromHeader).ToList(); + + if (!headerParameters.Any()) return true; + foreach (var item in headerParameters) + { + var p = item.p; + if (p.Type is INamedTypeSymbol + { + TypeKind: TypeKind.Class, SpecialType: not SpecialType.System_String + } parameterClassType) + { + var properties = parameterClassType.GetMembers().Where(m => m.Kind == SymbolKind.Property); + foreach (var prop in properties) + { + statements.Add( + $$"""_request_gen.Headers.Add(nameof({{p.Name}}.{{prop.Name}}), $"{{{p.Name}}.{{prop.Name}}}")"""); + } + } + else + { + statements.Add($$"""_request_gen.Headers.Add(nameof({{p.Name}}), $"{{{p.Name}}}")"""); + } + } + + return true; + } + + private static bool AddResponseHandling(List statements, ITypeSymbol returnType, List errorAndWarnings, string? cancellationTokenName) + { + cancellationTokenName ??= string.Empty; + if (returnType is { TypeKind: TypeKind.Class, SpecialType: not SpecialType.System_String }) + { + statements.Add($"var _stream_gen = await _response_gen.Content.ReadAsStreamAsync({cancellationTokenName})"); + statements.Add( + $"return global::System.Text.Json.JsonSerializer.Deserialize<{returnType.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)}>(_stream_gen, _JSON_OPTIONS_gen);"); + } + else + { + statements.Add($"var _str_gen = await _response_gen.Content.ReadAsStringAsync({cancellationTokenName})"); + if (returnType.SpecialType == SpecialType.System_String) + { + statements.Add("return _str_gen"); + } + else if (returnType.IsValueType) + { + statements.Add($"{returnType.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)}.TryParse(_str_gen, out var val)"); + statements.Add("return val"); + } + else + { + errorAndWarnings.Add(DiagnosticDefinitions.WAG00009(returnType.Locations.FirstOrDefault())); + return false; + } + } + + return true; + } + + + private static ConstructorBuilder BuildConstructor(INamedTypeSymbol classSymbol, bool needAuth) + { + List parameters = ["global::System.Net.Http.IHttpClientFactory factory"]; + List body = ["clientFactory = factory;"]; + if (needAuth) + { + //parameters.Add("global::AutoWasmApiGenerator.IHttpClientHeaderHandler handler"); + parameters.Add("global::System.IServiceProvider services"); + body.Add( + "headerHandler = services.GetService() ?? global::AutoWasmApiGenerator.DefaultHttpClientHeaderHandler.Default"); + } + + return ConstructorBuilder.Default + .MethodName($"{FormatClassName(classSymbol.MetadataName)}ApiInvoker") + .AddParameter([.. parameters]) + .AddBody([.. body]) + .AddBody( + "_JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true };"); + } + + private static ClassBuilder CreateHttpClassBuilder(INamedTypeSymbol interfaceSymbol) + { + IEnumerable additionalAttribute = []; + if (interfaceSymbol.GetAttribute(ApiInvokerAttributeFullName, out var data)) + { + //var o = data.GetAttributeValue(nameof(ApiInvokerGeneraAttribute.Attribute)); + additionalAttribute = interfaceSymbol.GetAttributeInitInfo(ApiInvokerAttributeFullName, data!); + } + + + return ClassBuilder.Default + .ClassName($"{FormatClassName(interfaceSymbol.MetadataName)}ApiInvoker") + .AddGeneratedCodeAttribute(typeof(HttpServiceInvokerGenerator)) + .Attribute([.. additionalAttribute.Select(i => i.ToString())]) + .BaseType(interfaceSymbol.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)); + } +} diff --git a/src/AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs b/src/AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs new file mode 100644 index 0000000..bb06939 --- /dev/null +++ b/src/AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs @@ -0,0 +1,10 @@ +using System.Collections.Generic; +using Microsoft.CodeAnalysis; + +namespace AutoWasmApiGenerator; + +public interface IHttpServiceInvokerGenerator +{ + (string generatedFileName, string sourceCode, List errorAndWarnings) Generate( + INamedTypeSymbol interfaceSymbol); +} \ No newline at end of file From a46149c2c1c6ef61221c5780423f325fbca692a2 Mon Sep 17 00:00:00 2001 From: ihonliu Date: Sat, 7 Dec 2024 11:05:06 +0800 Subject: [PATCH 07/20] =?UTF-8?q?chore:=20=E6=9B=B4=E6=96=B0=E5=AD=90?= =?UTF-8?q?=E6=A8=A1=E5=9D=97?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/Shared | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Shared b/src/Shared index 559b46e..6240edd 160000 --- a/src/Shared +++ b/src/Shared @@ -1 +1 @@ -Subproject commit 559b46e1d4721bb665d35acc66fbc36536f7989e +Subproject commit 6240eddf29548dba0d32e6b3dee8d3d510e1f468 From c468a8d7c29cff2562cb13d961d17d0d5195b3dc Mon Sep 17 00:00:00 2001 From: ihonliu Date: Sun, 8 Dec 2024 02:13:29 +0800 Subject: [PATCH 08/20] =?UTF-8?q?refactor:=20=E4=BF=AE=E6=94=B9=E8=AF=8A?= =?UTF-8?q?=E6=96=AD=E5=88=86=E7=B1=BB?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md | 2 +- src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md index a0028c9..7a5204d 100644 --- a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md +++ b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md @@ -7,7 +7,7 @@ Rule ID | Category | Severity | Notes WAG00001 | AutoWasmApiGenerator.ControllerGenerator | Error | 继承多个接口需要指定接口标注[WebControllerAttribute] WAG00002 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 无法为该类型生成WebApi调用类,缺少接口 WAG00003 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 方法参数过多 -WAG00004 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 控制器(controller)不能包含泛型 +WAG00004 | AutoWasmApiGenerator.ControllerGenerator | Error | 控制器(controller)不能包含泛型 WAG00005 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 仅支持异步方法 WAG00006 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 路由中未包含路由参数({0}) WAG00007 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 不能同时设置[FromBody]和[FromForm]({0}) diff --git a/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs b/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs index cb6b642..21dd9e0 100644 --- a/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs +++ b/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs @@ -35,7 +35,7 @@ internal class DiagnosticDefinitions id: "WAG00004", title: "控制器(controller)不能包含泛型", messageFormat: "控制器(controller)不能包含泛型", - category: typeof(HttpServiceInvokerGenerator).FullName!, + category: typeof(ControllerGenerator).FullName!, defaultSeverity: DiagnosticSeverity.Error, isEnabledByDefault: true); From 1be6bd5a578c154ffee7311383cbb27681ec50be Mon Sep 17 00:00:00 2001 From: ihonliu Date: Sun, 8 Dec 2024 02:32:39 +0800 Subject: [PATCH 09/20] =?UTF-8?q?feat:=20=E6=96=B0=E5=A2=9E=E5=B1=9E?= =?UTF-8?q?=E6=80=A7=E4=BE=9D=E8=B5=96=E9=94=99=E8=AF=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../AnalyzerReleases.Unshipped.md | 1 + .../DiagnosticDefinitions.cs | 77 +++++++++++++++++++ .../HttpServiceInvokerGenerator.cs | 4 + 3 files changed, 82 insertions(+) diff --git a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md index 7a5204d..f7a0679 100644 --- a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md +++ b/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md @@ -14,3 +14,4 @@ WAG00007 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 不能同 WAG00008 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 不能设置多个[FromBody]({0}) WAG00009 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 暂不支持的返回值类型({0}) WAG00010 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | 生成服务调用器过程中发生错误: {0} +WAG00011 | AutoWasmApiGenerator.HttpServiceInvokerGenerator | Error | AutoWasmApiGenerator.ApiInvokerGenerateAttribute需要同时设置AutoWasmApiGenerator.WebControllerAttribute \ No newline at end of file diff --git a/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs b/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs index 21dd9e0..4a5e442 100644 --- a/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs +++ b/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs @@ -2,6 +2,7 @@ using System; using System.Collections.Generic; using System.Text; +// ReSharper disable InconsistentNaming namespace AutoWasmApiGenerator { @@ -89,54 +90,130 @@ internal class DiagnosticDefinitions isEnabledByDefault: true ); + private static readonly DiagnosticDescriptor Wag00011DiagDescriptor = new( + id: "WAG00011", + title: "属性依赖错误", + messageFormat: "AutoWasmApiGenerator.ApiInvokerGenerateAttribute需要同时设置AutoWasmApiGenerator.WebControllerAttribute", + category: typeof(HttpServiceInvokerGenerator).FullName!, + defaultSeverity: DiagnosticSeverity.Error, + isEnabledByDefault: true + ); + + /// + /// 继承多个接口需要指定接口标注[WebControllerAttribute] + /// + /// + /// public static Diagnostic WAG00001(Location? location) { return Diagnostic.Create(Wag00001DiagDescriptor, location); } + /// + /// 无法为该类型生成WebApi调用类,缺少接口 + /// + /// + /// public static Diagnostic WAG00002(Location? location) { return Diagnostic.Create(Wag00002DiagDescriptor, location); } + /// + /// 方法参数过多 + /// + /// + /// public static Diagnostic WAG00003(Location? location) { return Diagnostic.Create(Wag00003DiagDescriptor, location); } + /// + /// 控制器(controller)不能包含泛型 + /// + /// + /// + /// public static Diagnostic WAG00004(Location? location, string? symbolString = null) { return Diagnostic.Create(Wag00004DiagDescriptor, location, symbolString); } + /// + /// 仅支持异步方法 + /// + /// + /// + /// public static Diagnostic WAG00005(Location? location, string? symbolString = null) { return Diagnostic.Create(Wag00005DiagDescriptor, location, symbolString); } + /// + /// 路由中未包含路由参数 + /// + /// + /// + /// public static Diagnostic WAG00006(Location? location, string? symbolString = null) { return Diagnostic.Create(Wag00006DiagDescriptor, location, symbolString); } + /// + /// 不能同时设置[FromBody]和[FromForm] + /// + /// + /// + /// public static Diagnostic WAG00007(Location? location, string? symbolString = null) { return Diagnostic.Create(Wag00007DiagDescriptor, location, symbolString); } + /// + /// 不能设置多个[FromBody] + /// + /// + /// + /// public static Diagnostic WAG00008(Location? location, string? symbolString = null) { return Diagnostic.Create(Wag00008DiagDescriptor, location, symbolString); } + /// + /// 暂不支持的返回值类型 + /// + /// + /// + /// public static Diagnostic WAG00009(Location? location, string? symbolString = null) { return Diagnostic.Create(Wag00009DiagDescriptor, location, symbolString); } + /// + /// 生成错误 + /// + /// + /// + /// public static Diagnostic WAG00010(Location? location, string symbolString) { return Diagnostic.Create(Wag00010DiagDescriptor, location, symbolString); } + + /// + /// 属性依赖错误 + /// + /// + /// + public static Diagnostic WAG00011(Location? location) + { + return Diagnostic.Create(Wag00011DiagDescriptor, location); + } } } diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs index 9727610..cd22cac 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs @@ -27,6 +27,10 @@ public void Initialize(IncrementalGeneratorInitializationContext context) { if (!item.HasAttribute(WebControllerAttributeFullName)) { + foreach (var location in item.Locations) + { + context.ReportDiagnostic(DiagnosticDefinitions.WAG00011(location)); + } continue; } From 8b288fda62103af75f70007026907f41d9d1588b Mon Sep 17 00:00:00 2001 From: ihonliu Date: Sun, 8 Dec 2024 02:34:53 +0800 Subject: [PATCH 10/20] =?UTF-8?q?refactor:=20=E9=87=8D=E5=91=BD=E5=90=8D?= =?UTF-8?q?=E5=B8=B8=E9=87=8F=E5=90=8D=E7=A7=B0=EF=BC=8C=E7=A7=BB=E9=99=A4?= =?UTF-8?q?=E6=97=A0=E7=94=A8=E5=B1=9E=E6=80=A7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/AutoWasmApi.Roslyn/GeneratorHelpers.cs | 2 +- .../HttpServiceInvokerGenerator.cs | 3 ++- .../AutoWasmApiGenerator.csproj | 14 +++++++------- 3 files changed, 10 insertions(+), 9 deletions(-) diff --git a/src/AutoWasmApi.Roslyn/GeneratorHelpers.cs b/src/AutoWasmApi.Roslyn/GeneratorHelpers.cs index cc829a5..54d4fcb 100644 --- a/src/AutoWasmApi.Roslyn/GeneratorHelpers.cs +++ b/src/AutoWasmApi.Roslyn/GeneratorHelpers.cs @@ -8,7 +8,7 @@ public static class GeneratorHelpers { public const string WebControllerAttributeFullName = "AutoWasmApiGenerator.WebControllerAttribute"; public const string WebControllerAssemblyAttributeFullName = "AutoWasmApiGenerator.WebControllerAssemblyAttribute"; - public const string ApiInvokerAttributeFullName = "AutoWasmApiGenerator.ApiInvokerGenerateAttribute"; + public const string ApiInvokerGenerateAttributeFullName = "AutoWasmApiGenerator.ApiInvokerGenerateAttribute"; public const string ApiInvokerAssemblyAttributeFullName = "AutoWasmApiGenerator.ApiInvokerAssemblyAttribute"; public const string WebMethodAttributeFullName = "AutoWasmApiGenerator.WebMethodAttribute"; public const string DisableWebApiGenerator = "DisableWebApiGenerator"; diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs index cd22cac..23129f3 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs @@ -22,7 +22,8 @@ public void Initialize(IncrementalGeneratorInitializationContext context) return; } - var all = compilation.GetAllSymbols(ApiInvokerAttributeFullName); + var all = compilation.GetAllSymbols(ApiInvokerGenerateAttributeFullName); + foreach (var item in all) { if (!item.HasAttribute(WebControllerAttributeFullName)) diff --git a/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj index 7239835..e7d6faf 100644 --- a/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj +++ b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj @@ -17,15 +17,15 @@ - + - + - - - - - + + + + + From c11a1523ac38b90888088b6ad7a8fe86bfb12d14 Mon Sep 17 00:00:00 2001 From: ihonliu Date: Sun, 8 Dec 2024 02:36:57 +0800 Subject: [PATCH 11/20] =?UTF-8?q?refactor:=20=E4=BD=BF=E7=94=A8=E6=9E=9A?= =?UTF-8?q?=E4=B8=BE=E7=B1=BB=E5=9E=8B=E6=9B=BF=E4=BB=A3=E4=B9=8B=E5=89=8D?= =?UTF-8?q?=E7=9A=84=E5=B8=B8=E9=87=8F?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../HttpServiceInvokerGeneratorImpl.cs | 77 +++++++++++-------- 1 file changed, 46 insertions(+), 31 deletions(-) diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs index 6cd7095..98234aa 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs @@ -26,13 +26,15 @@ private static class WebMethodConstants public const string Post = "Post"; public const string Put = "Put"; public const string Delete = "Delete"; - - public const int Ignore = -1; - public const int FromQuery = 0; - public const int FromRoute = 1; - public const int FromForm = 2; - public const int FromBody = 3; - public const int FromHeader = 4; + } + private enum ParameterBindingType + { + Ignore = -1, + FromQuery = 0, + FromRoute = 1, + FromForm = 2, + FromBody = 3, + FromHeader = 4 } public (string generatedFileName, string sourceCode, List errorAndWarnings) Generate( @@ -92,7 +94,7 @@ private static bool CreateCodeFile(INamedTypeSymbol interfaceSymbol, List errorAndWarnings) { var methodSymbol = method.Item1; @@ -175,32 +177,34 @@ private static bool CreateCodeFile(INamedTypeSymbol interfaceSymbol, List t.Item1 == WebMethodConstants.FromRoute); - foreach ((int _, IParameterSymbol p) item in routerParameters) + var routerParameters = paramInfos.Where(t => t.bindingType == ParameterBindingType.FromRoute); + foreach (var item in routerParameters) { + // 如果路由参数中包含方法名,则忽略 if (methodRoute.Contains($"{{{item.p.Name}}}")) continue; errorAndWarnings.Add(DiagnosticDefinitions.WAG00006(methodSymbol.Locations.FirstOrDefault(), methodSymbol.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat))); return null; } - if (paramInfos.Any(t => t.Item1 == 2) && paramInfos.Any(t => t.Item1 == 3)) + if (paramInfos.Any(t => t.bindingType == ParameterBindingType.FromForm) && paramInfos.Any(t => t.bindingType == ParameterBindingType.FromBody)) { - errorAndWarnings.Add(DiagnosticDefinitions.WAG00006(methodSymbol.Locations.FirstOrDefault(), + // 不能同时存在FromBody和FromForm + errorAndWarnings.Add(DiagnosticDefinitions.WAG00007(methodSymbol.Locations.FirstOrDefault(), methodSymbol.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat))); return null; } @@ -272,9 +276,9 @@ private static IEnumerable BuildField(bool needAuth) /// /// private static void AddQueryParameters(List statements, - IEnumerable<(int i, IParameterSymbol p)> paramInfos, string webMethod, List errorAndWarnings) + IEnumerable<(ParameterBindingType i, IParameterSymbol p)> paramInfos, string webMethod, List errorAndWarnings) { - var queryParameters = paramInfos.Where(t => (t.i == -1 && webMethod == WebMethodConstants.Get) || t.i == WebMethodConstants.FromQuery).ToList(); + var queryParameters = paramInfos.Where(t => (t.i == ParameterBindingType.Ignore && webMethod == WebMethodConstants.Get) || t.i == ParameterBindingType.FromQuery).ToList(); if (!queryParameters.Any()) return; statements.Add("var _queries_gen = new global::System.Collections.Generic.List()"); @@ -310,9 +314,9 @@ private static void AddQueryParameters(List statements, /// /// /// - private static void AddFormParameters(List statements, IEnumerable<(int i, IParameterSymbol p)> paramInfos) + private static void AddFormParameters(List statements, IEnumerable<(ParameterBindingType i, IParameterSymbol p)> paramInfos) { - var formParameters = paramInfos.Where(t => t.i == WebMethodConstants.FromForm).ToList(); + var formParameters = paramInfos.Where(t => t.i == ParameterBindingType.FromForm).ToList(); if (!formParameters.Any()) return; statements.Add( @@ -354,32 +358,42 @@ private static void AddFormParameters(List statements, IEnumerable<(i /// /// /// - private static bool AddBodyParameters(List statements, IEnumerable<(int i, IParameterSymbol p)> paramInfos, string webMethod, IMethodSymbol methodSymbol, List errorAndWarnings) + private static bool AddBodyParameters(List statements, IEnumerable<(ParameterBindingType i, IParameterSymbol p)> paramInfos, string webMethod, IMethodSymbol methodSymbol, List errorAndWarnings) { // var bodyParameters = paramInfos.Where(t => (t.i == -1 && webMethod != WebMethodConstants.Get) || t.i == WebMethodConstants.Body).ToList(); - var bodyParameters = paramInfos.Where(t => t.i == WebMethodConstants.FromBody).ToList(); - if (bodyParameters.Count > 1) + var bodyParameters = paramInfos.Where(t => t.i == ParameterBindingType.FromBody).ToList(); + return bodyParameters.Count switch + { + > 1 => DoReturnError(), + 0 => true, + 1 => DoAddBody(), + _ => throw new ArgumentOutOfRangeException() + }; + + bool DoReturnError() { errorAndWarnings.Add(DiagnosticDefinitions.WAG00008(methodSymbol.Locations.FirstOrDefault())); return false; } - if (bodyParameters.Count == 0) return true; - var p = bodyParameters.First().p; + bool DoAddBody() + { + var p = bodyParameters[0].p; statements.Add($"var _json_gen = global::System.Text.Json.JsonSerializer.Serialize({p.Name})"); statements.Add( """_request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json")"""); return true; } + } /// /// 处理Header参数 /// /// /// - private static bool AddHeaderParameters(List statements, IEnumerable<(int i, IParameterSymbol p)> paramInfos) + private static bool AddHeaderParameters(List statements, IEnumerable<(ParameterBindingType i, IParameterSymbol p)> paramInfos) { - var headerParameters = paramInfos.Where(t => t.Item1 == WebMethodConstants.FromHeader).ToList(); + var headerParameters = paramInfos.Where(t => t.i == ParameterBindingType.FromHeader).ToList(); if (!headerParameters.Any()) return true; foreach (var item in headerParameters) @@ -406,7 +420,8 @@ private static bool AddHeaderParameters(List statements, IEnumerable< return true; } - private static bool AddResponseHandling(List statements, ITypeSymbol returnType, List errorAndWarnings, string? cancellationTokenName) + private static bool AddResponseHandling(List statements, ITypeSymbol returnType, + List errorAndWarnings, string? cancellationTokenName, IMethodSymbol methodSymbol) { cancellationTokenName ??= string.Empty; if (returnType is { TypeKind: TypeKind.Class, SpecialType: not SpecialType.System_String }) @@ -429,7 +444,7 @@ private static bool AddResponseHandling(List statements, ITypeSymbol } else { - errorAndWarnings.Add(DiagnosticDefinitions.WAG00009(returnType.Locations.FirstOrDefault())); + errorAndWarnings.Add(DiagnosticDefinitions.WAG00009(methodSymbol.Locations.FirstOrDefault(), returnType.ToDisplayString(SymbolDisplayFormat.CSharpErrorMessageFormat))); return false; } } @@ -461,10 +476,10 @@ private static ConstructorBuilder BuildConstructor(INamedTypeSymbol classSymbol, private static ClassBuilder CreateHttpClassBuilder(INamedTypeSymbol interfaceSymbol) { IEnumerable additionalAttribute = []; - if (interfaceSymbol.GetAttribute(ApiInvokerAttributeFullName, out var data)) + if (interfaceSymbol.GetAttribute(ApiInvokerGenerateAttributeFullName, out var data)) { //var o = data.GetAttributeValue(nameof(ApiInvokerGeneraAttribute.Attribute)); - additionalAttribute = interfaceSymbol.GetAttributeInitInfo(ApiInvokerAttributeFullName, data!); + additionalAttribute = interfaceSymbol.GetAttributeInitInfo(ApiInvokerGenerateAttributeFullName, data!); } From 0e5c66a0f711c99289253daca39268e3c05c91cd Mon Sep 17 00:00:00 2001 From: ihonliu Date: Sun, 8 Dec 2024 02:41:59 +0800 Subject: [PATCH 12/20] =?UTF-8?q?refactor:=20=E4=BF=AE=E6=94=B9=E8=BF=94?= =?UTF-8?q?=E5=9B=9E=E7=B1=BB=E5=9E=8B=E6=98=AF=E5=90=A6=E4=BD=BF=E7=94=A8?= =?UTF-8?q?tryparse=E7=9A=84=E9=80=BB=E8=BE=91?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Extensions/MethodCheckerExtensions.cs | 17 +++++++++++++++++ .../HttpServiceInvokerGeneratorImpl.cs | 3 ++- 2 files changed, 19 insertions(+), 1 deletion(-) create mode 100644 src/AutoWasmApi.Roslyn/Extensions/MethodCheckerExtensions.cs diff --git a/src/AutoWasmApi.Roslyn/Extensions/MethodCheckerExtensions.cs b/src/AutoWasmApi.Roslyn/Extensions/MethodCheckerExtensions.cs new file mode 100644 index 0000000..8106e10 --- /dev/null +++ b/src/AutoWasmApi.Roslyn/Extensions/MethodCheckerExtensions.cs @@ -0,0 +1,17 @@ +using Microsoft.CodeAnalysis; + +namespace AutoWasmApiGenerator.Extensions; + +internal static class MethodCheckerExtensions +{ + public static bool HasTryParseMethod(this ITypeSymbol returnType) + { + foreach (var method in returnType.GetMembers("TryParse").OfType()) + { + var match = method is { Parameters.Length: 2, ReturnType.Name: "Boolean" }; + if (match) return true; + } + + return false; + } +} \ No newline at end of file diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs index 98234aa..bf23d01 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs @@ -5,6 +5,7 @@ using System.Linq; using System.Text.RegularExpressions; using System.Threading; +using AutoWasmApiGenerator.Extensions; using Generators.Models; using Generators.Shared; using Generators.Shared.Builder; @@ -437,7 +438,7 @@ private static bool AddResponseHandling(List statements, ITypeSymbol { statements.Add("return _str_gen"); } - else if (returnType.IsValueType) + else if (returnType.HasTryParseMethod()) { statements.Add($"{returnType.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)}.TryParse(_str_gen, out var val)"); statements.Add("return val"); From ba6fc6e6c20e2d60148f635133b853efaa83443c Mon Sep 17 00:00:00 2001 From: ihonliu Date: Sun, 8 Dec 2024 02:43:31 +0800 Subject: [PATCH 13/20] =?UTF-8?q?chore:=20=E6=B8=85=E7=90=86=20&=20?= =?UTF-8?q?=E4=BF=AE=E6=94=B9=E5=AD=97=E7=AC=A6=E4=B8=B2=E5=86=99=E6=B3=95?= =?UTF-8?q?=20&=20=E6=9B=B4=E6=96=B0=E6=96=87=E6=A1=A3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../AutoWasmApi.Roslyn.csproj | 4 ---- src/AutoWasmApi.Roslyn/ControllerGenerator.cs | 6 ----- .../HttpServiceInvokerGeneratorImpl.cs | 23 +++++++++---------- src/AutoWasmApiGenerator/readme.md | 11 +++++++++ 4 files changed, 22 insertions(+), 22 deletions(-) diff --git a/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj b/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj index 7b2a2e3..4ad31fb 100644 --- a/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj +++ b/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj @@ -9,9 +9,5 @@ - - - - diff --git a/src/AutoWasmApi.Roslyn/ControllerGenerator.cs b/src/AutoWasmApi.Roslyn/ControllerGenerator.cs index 8db296b..6e529fc 100644 --- a/src/AutoWasmApi.Roslyn/ControllerGenerator.cs +++ b/src/AutoWasmApi.Roslyn/ControllerGenerator.cs @@ -25,12 +25,6 @@ public void Initialize(IncrementalGeneratorInitializationContext context) // WebControllerAttributeFullName // , static (node, _) => true // , static (ctx, _) => ctx); -#if DEBUG && false - if (!Debugger.IsAttached) - { - Debugger.Launch(); // This will launch the debugger when the source generator runs. - } -#endif context.RegisterSourceOutput(context.CompilationProvider, static (context, compilation) => { diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs index bf23d01..a50b5ab 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs @@ -68,7 +68,6 @@ private static bool CreateCodeFile(INamedTypeSymbol interfaceSymbol, List 0) { - // Debug.Assert(methodBuilder == null); file = null; return false; } @@ -86,10 +85,11 @@ private static bool CreateCodeFile(INamedTypeSymbol interfaceSymbol, List", -"#pragma warning disable", -"#nullable enable") + .AddFileHeader(""" + // + #pragma warning disable + #nullable enable + """) .AddMembers(namespaceBuilder); return true; @@ -237,7 +237,7 @@ private static bool CreateCodeFile(INamedTypeSymbol interfaceSymbol, List $"{p.Type.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)} {p.Name}").ToArray(); @@ -380,11 +380,11 @@ bool DoReturnError() bool DoAddBody() { var p = bodyParameters[0].p; - statements.Add($"var _json_gen = global::System.Text.Json.JsonSerializer.Serialize({p.Name})"); - statements.Add( - """_request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json")"""); - return true; - } + statements.Add($"var _json_gen = global::System.Text.Json.JsonSerializer.Serialize({p.Name})"); + statements.Add( + """_request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json")"""); + return true; + } } /// @@ -483,7 +483,6 @@ private static ClassBuilder CreateHttpClassBuilder(INamedTypeSymbol interfaceSym additionalAttribute = interfaceSymbol.GetAttributeInitInfo(ApiInvokerGenerateAttributeFullName, data!); } - return ClassBuilder.Default .ClassName($"{FormatClassName(interfaceSymbol.MetadataName)}ApiInvoker") .AddGeneratedCodeAttribute(typeof(HttpServiceInvokerGenerator)) diff --git a/src/AutoWasmApiGenerator/readme.md b/src/AutoWasmApiGenerator/readme.md index 0a319ee..6573127 100644 --- a/src/AutoWasmApiGenerator/readme.md +++ b/src/AutoWasmApiGenerator/readme.md @@ -111,3 +111,14 @@ Task Log3Async([WebMethodParameterBinding(BindingType.FromBody)] string me + FromBody 从请求正文中获取值。 + FromHeader 从 HTTP 标头中获取值。 + FromServices 从服务容器中获取值。 + + +## TODOs + +- [ ] 支持类partial功能 +- [ ] 支持方法partial功能 +- [ ] 支持方法名称自定义功能 +- [ ] 支持同步方法 +- [ ] 支持同时设置`[FromBody]`和`[FromForm]` (需要参考一下此功能是否可以实现) +- [ ] 增加全局配置 +- [ ] 整理Analyzer中的错误提示 From 65f8a540a36350bd52296503513a4d7725135453 Mon Sep 17 00:00:00 2001 From: ihonliu Date: Sun, 8 Dec 2024 03:02:33 +0800 Subject: [PATCH 14/20] =?UTF-8?q?feat:=20=E5=A2=9E=E5=8A=A0=E5=8D=95?= =?UTF-8?q?=E5=85=83=E6=B5=8B=E8=AF=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../AutoWasmApiGenerator.Test.csproj | 55 +++ ...tpServiceInvokerGeneratorTest.Constants.cs | 372 ++++++++++++++++++ .../HttpServiceInvokerGeneratorTest.cs | 275 +++++++++++++ .../IncrementalSourceGeneratorTestBase.cs | 22 ++ src/MT.Generators.sln | 24 +- 5 files changed, 740 insertions(+), 8 deletions(-) create mode 100644 src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj create mode 100644 src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs create mode 100644 src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs create mode 100644 src/AutoWasmApiGenerator.Test/IncrementalSourceGeneratorTestBase.cs diff --git a/src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj b/src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj new file mode 100644 index 0000000..8f8c904 --- /dev/null +++ b/src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj @@ -0,0 +1,55 @@ + + + + net8.0 + enable + enable + + false + true + + + + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + + + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + + + + + + + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + + + + + + + + + + + + + + diff --git a/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs b/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs new file mode 100644 index 0000000..e33370e --- /dev/null +++ b/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs @@ -0,0 +1,372 @@ +// ReSharper disable InconsistentNaming +namespace AutoWasmApiGenerator.Test; + +public partial class HttpServiceInvokerGeneratorTest +{ + private const string AssemblyTag = @" [assembly: AutoWasmApiGenerator.ApiInvokerAssembly] + [assembly: AutoWasmApiGenerator.WebControllerAssembly] +"; + + private const string Usings = @"global using AutoWasmApiGenerator; +global using System; +global using System.Threading; +global using System.Threading.Tasks; +global using System.Text.Json; +global using System.Net.Http; +global using System.Collections.Generic; +global using System.Text; +global using System.Linq; +// using Microsoft.Extensions.DependencyInjection; +"; + + #region Wag00005 + + private const string TestWag00005 = @"namespace AutoWasmApiGenerator.Test; + +[WebController(Authorize = true)] +[ApiInvokerGenerate] +public interface ITest +{ + void Log(string message); +}"; + + #endregion + + #region Wag00006 + + private const string TestWag00006 = @"using AutoWasmApiGenerator +namespace AutoWasmApiGenerator.Test; + +[WebController] +[ApiInvokerGenerate] +public interface ITest +{ + [WebMethod(Method = WebMethod.Post)] + Task LogAsync([WebMethodParameterBinding(BindingType.FromRoute)] string model0)] TestModel1 model1); +} +"; + + #endregion + + #region Wag00007 + + private const string TestWag00007 = @"using AutoWasmApiGenerator +namespace AutoWasmApiGenerator.Test; + +public record TestModel0(string Name, int Age); + +public record TestModel1(string Name, double Salary); + +[WebController] +[ApiInvokerGenerate] +public interface ITest +{ + [WebMethod(Method = WebMethod.Post)] + Task LogAsync([WebMethodParameterBinding(BindingType.FromBody)] TestModel0 model0, [WebMethodParameterBinding(BindingType.FromForm)] TestModel1 model1); +} + +"; + + #endregion + + #region Wag00010 + + private const string TestWag00010 = @"using AutoWasmApiGenerator"; + // private const string TestWag00010Result = @"using AutoWasmApiGenerator"; + + #endregion + + #region Wag00011 + + private const string TestWag00011 = @"namespace AutoWasmApiGenerator.Test; + +[ApiInvokerGenerate] +public interface ITest +{ + [WebMethod(Method = WebMethod.Get)] + Task LogAsync(string message); +}"; + + #endregion + + #region Success + + private const string TestSuccessCode = @"using AutoWasmApiGenerator; +using System.Threading.Tasks; +namespace AutoWasmApiGenerator.Test; + +[WebController(Authorize = true)] +[ApiInvokerGenerate] +public interface ITest +{ + [ApiInvokeNotSupported] + void Log(string message); + [WebMethod(Method = WebMethod.Get)] + Task LogAsync(string message); + + [WebMethod(Method = WebMethod.Post)] + Task Log2Async([WebMethodParameterBinding(BindingType.FromBody)] string message, [WebMethodParameterBinding(BindingType.Ignore)] CancellationToken token); + + [WebMethod(Method = WebMethod.Delete)] + Task Log3Async([WebMethodParameterBinding(BindingType.FromBody)] string message, [WebMethodParameterBinding(BindingType.FromQuery)] string path, CancellationToken token); +} +"; + + private const string TestSuccessResult = + """ + // + #pragma warning disable + #nullable enable + + using Microsoft.Extensions.DependencyInjection; + + namespace AutoWasmApiGenerator.Test + { + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] + /// + public partial class TestApiInvoker : global::AutoWasmApiGenerator.Test.ITest + { + private readonly global::System.Text.Json.JsonSerializerOptions _JSON_OPTIONS_gen; + + private readonly global::System.Net.Http.IHttpClientFactory clientFactory; + + private readonly global::AutoWasmApiGenerator.IHttpClientHeaderHandler headerHandler; + + public TestApiInvoker(global::System.Net.Http.IHttpClientFactory factory, global::System.IServiceProvider services) + { + clientFactory = factory; + headerHandler = services.GetService() ?? global::AutoWasmApiGenerator.DefaultHttpClientHeaderHandler.Default; + _JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true }; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] + public void Log(string message) + => throw new global::System.NotSupportedException(); + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] + public async global::System.Threading.Tasks.Task LogAsync(string message) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Get; + await headerHandler.SetRequestHeaderAsync(_request_gen, global::System.Threading.CancellationToken.None); + var _url_gen = $"api/Test/Log"; + var _queries_gen = new global::System.Collections.Generic.List(); + _queries_gen.Add($"{nameof(message)}={message}"); + _url_gen = $"{_url_gen}?{string.Join("&", _queries_gen)}"; + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(); + bool.TryParse(_str_gen, out var val); + return val; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] + public async global::System.Threading.Tasks.Task Log2Async(string message, global::System.Threading.CancellationToken token) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Post; + await headerHandler.SetRequestHeaderAsync(_request_gen, token); + var _url_gen = $"api/Test/Log2"; + var _json_gen = global::System.Text.Json.JsonSerializer.Serialize(message); + _request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json"); + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen, token); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(token); + bool.TryParse(_str_gen, out var val); + return val; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] + public async global::System.Threading.Tasks.Task Log3Async(string message, string path, global::System.Threading.CancellationToken token) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Delete; + await headerHandler.SetRequestHeaderAsync(_request_gen, token); + var _url_gen = $"api/Test/Log3"; + var _queries_gen = new global::System.Collections.Generic.List(); + _queries_gen.Add($"{nameof(path)}={path}"); + _url_gen = $"{_url_gen}?{string.Join("&", _queries_gen)}"; + var _json_gen = global::System.Text.Json.JsonSerializer.Serialize(message); + _request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json"); + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen, token); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(token); + bool.TryParse(_str_gen, out var val); + return val; + } + } + } + """; + + #endregion + + #region Wag00002 + + private const string TestWag00002 = @"using AutoWasmApiGenerator"; + private const string TestWag00002Result = @"using AutoWasmApiGenerator"; + + #endregion + + #region Wag00003 + + private const string TestWag00003 = @"using AutoWasmApiGenerator"; + private const string TestWag00003Result = @"using AutoWasmApiGenerator"; + + #endregion + + #region FromRoute + + private const string TestFromRoute = + """ + using AutoWasmApiGenerator + namespace AutoWasmApiGenerator.Test; + + [WebController] + [ApiInvokerGenerate] + public interface ITest + { + [WebMethod(Method = WebMethod.Post, Route="Log/{model0}")] + Task LogAsync([WebMethodParameterBinding(BindingType.FromRoute)] string model0); + } + """; + + private const string TestFromRouteResult = + """ + // + #pragma warning disable + #nullable enable + + using Microsoft.Extensions.DependencyInjection; + + namespace AutoWasmApiGenerator.Test + { + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] + /// + public partial class TestApiInvoker : global::AutoWasmApiGenerator.Test.ITest + { + private readonly global::System.Text.Json.JsonSerializerOptions _JSON_OPTIONS_gen; + + private readonly global::System.Net.Http.IHttpClientFactory clientFactory; + + public TestApiInvoker(global::System.Net.Http.IHttpClientFactory factory) + { + clientFactory = factory; + _JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true }; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] + public async global::System.Threading.Tasks.Task LogAsync(string model0) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Post; + var _url_gen = $"api/Test/Log/Log/{model0}"; + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(); + bool.TryParse(_str_gen, out var val); + return val; + } + } + } + """; + + #endregion + + #region Wag00008 + + private const string TestWag00008 = @"using AutoWasmApiGenerator +namespace AutoWasmApiGenerator.Test; + +public record TestModel0(string Name, int Age); + +public record TestModel1(string Name, double Salary); + +[WebController] +[ApiInvokerGenerate] +public interface ITest +{ + [WebMethod(Method = WebMethod.Post)] + Task LogAsync([WebMethodParameterBinding(BindingType.FromBody)] TestModel0 model0, [WebMethodParameterBinding(BindingType.FromBody)] TestModel1 model1); +} +"; + + private const string TestWag00008Result = @"using AutoWasmApiGenerator"; + + #endregion + + #region Wag00009 + + private const string TestWag00009_0 = @"namespace AutoWasmApiGenerator.Test; + +[WebController(Authorize = true)] +[ApiInvokerGenerate] +public interface ITest +{ + Task Log(string message); +}"; + + private const string TestWag00009_ValidReturnType = @"namespace AutoWasmApiGenerator.Test; + +[WebController(Authorize = true)] +[ApiInvokerGenerate] +public interface ITest +{ + Task Log(string message); +}"; + + private const string TestWag00009_ValidReturnTypeResult = + """ + // + #pragma warning disable + #nullable enable + + using Microsoft.Extensions.DependencyInjection; + + namespace AutoWasmApiGenerator.Test + { + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] + /// + public partial class TestApiInvoker : global::AutoWasmApiGenerator.Test.ITest + { + private readonly global::System.Text.Json.JsonSerializerOptions _JSON_OPTIONS_gen; + + private readonly global::System.Net.Http.IHttpClientFactory clientFactory; + + private readonly global::AutoWasmApiGenerator.IHttpClientHeaderHandler headerHandler; + + public TestApiInvoker(global::System.Net.Http.IHttpClientFactory factory, global::System.IServiceProvider services) + { + clientFactory = factory; + headerHandler = services.GetService() ?? global::AutoWasmApiGenerator.DefaultHttpClientHeaderHandler.Default; + _JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true }; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] + public async global::System.Threading.Tasks.Task Log(string message) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Post; + await headerHandler.SetRequestHeaderAsync(_request_gen, global::System.Threading.CancellationToken.None); + var _url_gen = $"api/Test/Log"; + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(); + bool.TryParse(_str_gen, out var val); + return val; + } + } + } + """; + + #endregion +} \ No newline at end of file diff --git a/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs b/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs new file mode 100644 index 0000000..c2f8253 --- /dev/null +++ b/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs @@ -0,0 +1,275 @@ +using FluentAssertions; +using Generators.Shared; +using Microsoft.CodeAnalysis; + +namespace AutoWasmApiGenerator.Test; + +public static class ExecutableReferenceExtension +{ + public static PortableExecutableReference[] DistinctReferences(this IEnumerable types) + { + return types.Select(t => t.Assembly.Location).Distinct().Select(t => MetadataReference.CreateFromFile(t)) + .ToArray(); + } +} + +public partial class HttpServiceInvokerGeneratorTest : IncrementalSourceGeneratorTestBase +{ + private static readonly PortableExecutableReference[] References; + + static HttpServiceInvokerGeneratorTest() + { + References = new List + { + typeof(HttpServiceInvokerGenerator), + typeof(WebControllerAttribute) + }.DistinctReferences(); + } + + [Fact] + public void Test_ReferenceValidity() + { + var (compilation, _) = CreateDriver([TestSuccessCode, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + + var ret = compilation.GetAllSymbols(typeof(WebControllerAttribute).FullName!); + ret.Should().HaveCount(1); + } + + [Fact] + public void Test_SuccessGenerate() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestSuccessCode, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + + // Run the generator + driver = driver.RunGenerators(compilation); + var results = driver.GetRunResult(); + var result = results.Results.Single(); + + // Assert that the generated sources are not empty + Assert.NotEmpty(result.GeneratedSources); + result.GeneratedSources.Should().HaveCount(1).And.Contain(r => r.HintName == "AutoWasmApiGenerator_Test_TestApiInvoker.g.cs"); + result.GeneratedSources[0].SourceText.ToString().ReplaceLineEndings().Should() + .BeEquivalentTo(TestSuccessResult.ReplaceLineEndings()); + } + + [Fact] + public void Test_Wag00002() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestWag00002, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + // TODO: Add the test, however current error code is not used + } + + /// + /// ԣ + /// + [Fact] + public void Test_Wag00003() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestWag00003, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + // TODO: Add the test, however current error code is not used + } + + /// + /// ԣ֧첽 + /// + [Fact] + public void Test_Wag00005() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestWag00005, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + var result = driver.RunGenerators(compilation) + .GetRunResult(); + // result.Results.Should().BeEmpty(); + result.GeneratedTrees.Should().BeEmpty(); + result.Diagnostics.Should().HaveCount(1); + result.Diagnostics.Should().Contain(diagnostic => diagnostic.Id == "WAG00005"); + } + + /// + /// ԣ·δ·ɲ + /// + /// + [Fact] + public void Test_Wag00006_Fail() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestWag00006, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + var results = driver.RunGenerators(compilation) + .GetRunResult(); + results.Diagnostics.Should() + .HaveCount(1) + .And.Contain(diagnostic => diagnostic.Id == "WAG00006"); + results.GeneratedTrees.Should().BeEmpty(); + } + + /// + /// ԣ·δ·ɲ + /// + /// + [Fact] + public void Test_FromRoute() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestFromRoute, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + var results = driver.RunGenerators(compilation) + .GetRunResult(); + results.Diagnostics.Should().BeEmpty(); + results.GeneratedTrees.Should().NotBeEmpty(); + results.Results.Should().HaveCount(1); + var result = results.Results[0]; + result.GeneratedSources.Should().HaveCount(1); + string generated = result.GeneratedSources[0].SourceText.ToString().ReplaceLineEndings(); + generated.Should().BeEquivalentTo(TestFromRouteResult.ReplaceLineEndings()); + } + + + [Fact] + public void Test_Wag00007() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestWag00007, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + var results = driver.RunGenerators(compilation) + .GetRunResult(); + results.Diagnostics.Should() + .HaveCount(1) + .And.Contain(diagnostic => diagnostic.Id == "WAG00007"); + results.GeneratedTrees.Should().BeEmpty(); + } + + /// + /// ԣö[FromBody] + /// + /// + [Fact] + public void Test_Wag00008() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestWag00008, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + var results = driver.RunGenerators(compilation) + .GetRunResult(); + results.Diagnostics.Should() + .HaveCount(1) + .And.Contain(diagnostic => diagnostic.Id == "WAG00008"); + results.GeneratedTrees.Should().BeEmpty(); + } + + /// + /// ԣݲֵ֧ķֵ + /// + [Fact] + public void Test_Wag00009() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestWag00009_0, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + var results = driver.RunGenerators(compilation) + .GetRunResult(); + results.Diagnostics.Should() + .HaveCount(1) + .And.Contain(diagnostic => diagnostic.Id == "WAG00009"); + results.GeneratedTrees.Should().BeEmpty(); + + // Test valid return type, wag00009 should not be triggered + (compilation, driver) = CreateDriver([TestWag00009_ValidReturnType, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + + results = driver.RunGenerators(compilation).GetRunResult(); + results.Diagnostics.Should() + .BeEmpty(); + results.Results.Should().HaveCount(1); + var result = results.Results[0]; + + result.GeneratedSources.Should().HaveCount(1); + result.GeneratedSources[0].SourceText.ToString().ReplaceLineEndings().Should() + .BeEquivalentTo(TestWag00009_ValidReturnTypeResult.ReplaceLineEndings()); + } + + [Fact] + public void Test_Wag00010() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestWag00010, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + // TODO: Add the test, however current error code may not be triggered + } + + [Fact] + public void Test_Wag00011() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([TestWag00011, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + var results = driver.RunGenerators(compilation) + .GetRunResult(); + results.Diagnostics.Should() + .HaveCount(1) + .And.Contain(diagnostic => diagnostic.Id == "WAG00011"); + results.GeneratedTrees.Should().BeEmpty(); + } + + + #region Example + + /* // this just an example snippet, not a real test + public void Test_Example() + { + // Get AssemblyMeta from AutoWasmApiGenerator + var (compilation, driver) = CreateDriver([Test0, AssemblyTag, Usings], + new HttpServiceInvokerGenerator().AsSourceGenerator(), References); + + // Run the generator + driver = driver.RunGenerators(compilation); + var results = driver.GetRunResult(); + var result = results.Results.Single(); + // + // Log the diagnostics + foreach (var diagnostic in result.Diagnostics) + { + Console.WriteLine(diagnostic.ToString()); + } + + // Log the generated sources + foreach (var generatedSource in result.GeneratedSources) + { + Console.WriteLine($"HintName: {generatedSource.HintName}"); + Console.WriteLine(generatedSource.SourceText.ToString()); + } + + // Assert that the generated sources are not empty + Assert.NotEmpty(result.GeneratedSources); + + // Update the compilation and rerun the generator + compilation = compilation.AddSyntaxTrees(CSharpSyntaxTree.ParseText("// dummy")); + driver = driver.RunGenerators(compilation); + + // Assert the driver doesn't recompute the output + results = driver.GetRunResult(); + result = results.Results.Single(); + var allOutputs = result.TrackedOutputSteps.SelectMany(outputStep => outputStep.Value) + .SelectMany(output => output.Outputs); + Assert.Collection(allOutputs, output => Assert.Equal(IncrementalStepRunReason.Cached, output.Reason)); + + // Assert the driver use the cached result from AssemblyName and Syntax + var assemblyNameOutputs = result.TrackedSteps["AssemblyName"].Single().Outputs; + Assert.Collection(assemblyNameOutputs, + output => Assert.Equal(IncrementalStepRunReason.Unchanged, output.Reason)); + + var syntaxOutputs = result.TrackedSteps["Syntax"].Single().Outputs; + Assert.Collection(syntaxOutputs, output => Assert.Equal(IncrementalStepRunReason.Unchanged, output.Reason)); + } + */ + + #endregion +} \ No newline at end of file diff --git a/src/AutoWasmApiGenerator.Test/IncrementalSourceGeneratorTestBase.cs b/src/AutoWasmApiGenerator.Test/IncrementalSourceGeneratorTestBase.cs new file mode 100644 index 0000000..f9dc3d8 --- /dev/null +++ b/src/AutoWasmApiGenerator.Test/IncrementalSourceGeneratorTestBase.cs @@ -0,0 +1,22 @@ +using Basic.Reference.Assemblies; +using Microsoft.CodeAnalysis; +using Microsoft.CodeAnalysis.CSharp; + +namespace AutoWasmApiGenerator.Test; + +public abstract class IncrementalSourceGeneratorTestBase +{ + protected static (Compilation compilation, GeneratorDriver driver) CreateDriver(string[] source, ISourceGenerator sourceGenerator, params PortableExecutableReference[] references) + { + + var compilation = CSharpCompilation.Create("TestProject", + source.Select(t => CSharpSyntaxTree.ParseText(t)), + NetStandard20.References.All.AddRange(references), + new CSharpCompilationOptions(OutputKind.DynamicallyLinkedLibrary)); + // Create the generator driver + var driver = CSharpGeneratorDriver.Create( + [sourceGenerator], + driverOptions: new GeneratorDriverOptions(disabledOutputs: IncrementalGeneratorOutputKind.None, trackIncrementalGeneratorSteps: true)); + return (compilation, driver); + } +} \ No newline at end of file diff --git a/src/MT.Generators.sln b/src/MT.Generators.sln index f6c2c61..3a1d0a7 100644 --- a/src/MT.Generators.sln +++ b/src/MT.Generators.sln @@ -17,8 +17,6 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AutoInject.Roslyn", "AutoIn EndProject Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "AutoWasm", "AutoWasm", "{0D1AD616-22D2-4D10-BD23-92040AFD90D5}" EndProject -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AutoWasmApi.Roslyn", "AutoWasmApi.Roslyn\AutoWasmApi.Roslyn.csproj", "{625157FF-90BB-4950-9643-7BAEF5CEF6A8}" -EndProject Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Blazor.Test", "Blazor.Test\Blazor.Test\Blazor.Test.csproj", "{4A47713C-2332-40C1-9AAB-7B2A14D148A2}" EndProject Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Blazor.Test.Client", "Blazor.Test\Blazor.Test.Client\Blazor.Test.Client.csproj", "{20CA19C8-3036-4D1B-B095-C59A4C76B8D0}" @@ -43,6 +41,10 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Shared", "Shared", "{71D54F EndProject Project("{D954291E-2A0B-460D-934E-DC6B0785DB48}") = "Generators.Shared", "Shared\Generators.Shared\Generators.Shared.shproj", "{0C3325BB-C186-4E45-A567-69DAF5411E8B}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AutoWasmApiGenerator.Test", "AutoWasmApiGenerator.Test\AutoWasmApiGenerator.Test.csproj", "{3ED9CDA7-B256-4B4C-85DF-E1EA62715428}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AutoWasmApi.Roslyn", "AutoWasmApi.Roslyn\AutoWasmApi.Roslyn.csproj", "{144DCF75-901E-40EE-8449-FC404D57017D}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU @@ -69,10 +71,6 @@ Global {E48C1C4F-8111-4513-86EC-0C968E5437CD}.Debug|Any CPU.Build.0 = Debug|Any CPU {E48C1C4F-8111-4513-86EC-0C968E5437CD}.Release|Any CPU.ActiveCfg = Release|Any CPU {E48C1C4F-8111-4513-86EC-0C968E5437CD}.Release|Any CPU.Build.0 = Release|Any CPU - {625157FF-90BB-4950-9643-7BAEF5CEF6A8}.Debug|Any CPU.ActiveCfg = Debug|Any CPU - {625157FF-90BB-4950-9643-7BAEF5CEF6A8}.Debug|Any CPU.Build.0 = Debug|Any CPU - {625157FF-90BB-4950-9643-7BAEF5CEF6A8}.Release|Any CPU.ActiveCfg = Release|Any CPU - {625157FF-90BB-4950-9643-7BAEF5CEF6A8}.Release|Any CPU.Build.0 = Release|Any CPU {4A47713C-2332-40C1-9AAB-7B2A14D148A2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU {4A47713C-2332-40C1-9AAB-7B2A14D148A2}.Debug|Any CPU.Build.0 = Debug|Any CPU {4A47713C-2332-40C1-9AAB-7B2A14D148A2}.Release|Any CPU.ActiveCfg = Release|Any CPU @@ -101,6 +99,14 @@ Global {6FFF3104-CCD7-4916-A911-AEB9A5D35536}.Debug|Any CPU.Build.0 = Debug|Any CPU {6FFF3104-CCD7-4916-A911-AEB9A5D35536}.Release|Any CPU.ActiveCfg = Release|Any CPU {6FFF3104-CCD7-4916-A911-AEB9A5D35536}.Release|Any CPU.Build.0 = Release|Any CPU + {3ED9CDA7-B256-4B4C-85DF-E1EA62715428}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {3ED9CDA7-B256-4B4C-85DF-E1EA62715428}.Debug|Any CPU.Build.0 = Debug|Any CPU + {3ED9CDA7-B256-4B4C-85DF-E1EA62715428}.Release|Any CPU.ActiveCfg = Release|Any CPU + {3ED9CDA7-B256-4B4C-85DF-E1EA62715428}.Release|Any CPU.Build.0 = Release|Any CPU + {144DCF75-901E-40EE-8449-FC404D57017D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {144DCF75-901E-40EE-8449-FC404D57017D}.Debug|Any CPU.Build.0 = Debug|Any CPU + {144DCF75-901E-40EE-8449-FC404D57017D}.Release|Any CPU.ActiveCfg = Release|Any CPU + {144DCF75-901E-40EE-8449-FC404D57017D}.Release|Any CPU.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE @@ -110,7 +116,6 @@ Global {9C1A1324-0111-468F-9644-F80F72A66DC5} = {0D1AD616-22D2-4D10-BD23-92040AFD90D5} {1CA123D0-5B2A-4BBD-A90E-B58E01A7B2CA} = {FD7A81EE-6E54-461B-8766-BB36EB5F31D8} {E48C1C4F-8111-4513-86EC-0C968E5437CD} = {FD7A81EE-6E54-461B-8766-BB36EB5F31D8} - {625157FF-90BB-4950-9643-7BAEF5CEF6A8} = {0D1AD616-22D2-4D10-BD23-92040AFD90D5} {4A47713C-2332-40C1-9AAB-7B2A14D148A2} = {47FB0C42-25F4-49C1-A98C-D6536116B894} {20CA19C8-3036-4D1B-B095-C59A4C76B8D0} = {47FB0C42-25F4-49C1-A98C-D6536116B894} {8638056F-E36D-4EBC-A4EA-A7AE22DDC161} = {FCEAEFA9-132B-4945-9E97-53612B8A5CBF} @@ -119,14 +124,17 @@ Global {26DD2068-DD76-4FF9-8FB6-8E354188F7F7} = {B1852D0E-A185-491C-B81D-03D1AC5EF97C} {6FFF3104-CCD7-4916-A911-AEB9A5D35536} = {B1852D0E-A185-491C-B81D-03D1AC5EF97C} {0C3325BB-C186-4E45-A567-69DAF5411E8B} = {71D54FEF-E30B-4829-9EF2-C27DF766E235} + {3ED9CDA7-B256-4B4C-85DF-E1EA62715428} = {47FB0C42-25F4-49C1-A98C-D6536116B894} + {144DCF75-901E-40EE-8449-FC404D57017D} = {0D1AD616-22D2-4D10-BD23-92040AFD90D5} EndGlobalSection GlobalSection(ExtensibilityGlobals) = postSolution SolutionGuid = {E16846FC-C2B2-43ED-A534-5726B5C9DDEE} EndGlobalSection GlobalSection(SharedMSBuildProjectFiles) = preSolution Shared\Generators.Shared\Generators.Shared.projitems*{0c3325bb-c186-4e45-a567-69daf5411e8b}*SharedItemsImports = 13 + Shared\Generators.Shared\Generators.Shared.projitems*{144dcf75-901e-40ee-8449-fc404d57017d}*SharedItemsImports = 5 Shared\Generators.Shared\Generators.Shared.projitems*{26dd2068-dd76-4ff9-8fb6-8e354188f7f7}*SharedItemsImports = 5 - Shared\Generators.Shared\Generators.Shared.projitems*{625157ff-90bb-4950-9643-7baef5cef6a8}*SharedItemsImports = 5 + Shared\Generators.Shared\Generators.Shared.projitems*{3ed9cda7-b256-4b4c-85df-e1ea62715428}*SharedItemsImports = 5 Shared\Generators.Shared\Generators.Shared.projitems*{ba595e23-bd3a-4b36-a094-06e5814fe5e9}*SharedItemsImports = 5 Shared\Generators.Shared\Generators.Shared.projitems*{e48c1c4f-8111-4513-86ec-0c968e5437cd}*SharedItemsImports = 5 EndGlobalSection From 79568587749d6906c909171ddda7c29360b18d82 Mon Sep 17 00:00:00 2001 From: ihonliu Date: Sun, 8 Dec 2024 03:03:14 +0800 Subject: [PATCH 15/20] =?UTF-8?q?bug:=20=E7=A7=BB=E9=99=A4=E5=91=BD?= =?UTF-8?q?=E5=90=8D=E5=9F=9F=E4=B8=8A=E7=9A=84global::=E5=89=8D=E7=BC=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs index a50b5ab..bbc2491 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs @@ -81,7 +81,7 @@ private static bool CreateCodeFile(INamedTypeSymbol interfaceSymbol, List Date: Mon, 9 Dec 2024 17:45:48 +0800 Subject: [PATCH 16/20] =?UTF-8?q?chore:=20=E6=9B=B4=E6=96=B0=E5=AD=90?= =?UTF-8?q?=E6=A8=A1=E5=9D=97?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/Shared | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Shared b/src/Shared index 6240edd..e48adca 160000 --- a/src/Shared +++ b/src/Shared @@ -1 +1 @@ -Subproject commit 6240eddf29548dba0d32e6b3dee8d3d510e1f468 +Subproject commit e48adcac152e0a66651a03f5395b4023ec333247 From 03974124b6f289e49adf36a5e3a4b3c84ae933ed Mon Sep 17 00:00:00 2001 From: ihonliu Date: Tue, 10 Dec 2024 02:18:20 +0800 Subject: [PATCH 17/20] =?UTF-8?q?bug:=20=E4=BF=AE=E5=A4=8D=E6=9C=9F?= =?UTF-8?q?=E6=9C=9B=E6=96=87=E6=9C=AC=E4=B8=AD=E7=89=88=E6=9C=AC=E5=8F=B7?= =?UTF-8?q?=E4=B8=8D=E4=B8=80=E8=87=B4=E7=9A=84=E9=97=AE=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...tpServiceInvokerGeneratorTest.Constants.cs | 371 +++++++++--------- .../HttpServiceInvokerGeneratorTest.cs | 7 + 2 files changed, 199 insertions(+), 179 deletions(-) diff --git a/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs b/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs index e33370e..58f8520 100644 --- a/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs +++ b/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs @@ -89,8 +89,115 @@ public interface ITest #endregion + #region Wag00002 + + private const string TestWag00002 = @"using AutoWasmApiGenerator"; + + #endregion + + #region Wag00003 + + private const string TestWag00003 = @"using AutoWasmApiGenerator"; + + #endregion + #region Success + private static void UpdateSuccessTestString() + { + TestSuccessResult = + $$""" + // + #pragma warning disable + #nullable enable + + using Microsoft.Extensions.DependencyInjection; + + namespace AutoWasmApiGenerator.Test + { + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "{{GeneratorVersion}}")] + /// + public partial class TestApiInvoker : global::AutoWasmApiGenerator.Test.ITest + { + private readonly global::System.Text.Json.JsonSerializerOptions _JSON_OPTIONS_gen; + + private readonly global::System.Net.Http.IHttpClientFactory clientFactory; + + private readonly global::AutoWasmApiGenerator.IHttpClientHeaderHandler headerHandler; + + public TestApiInvoker(global::System.Net.Http.IHttpClientFactory factory, global::System.IServiceProvider services) + { + clientFactory = factory; + headerHandler = services.GetService() ?? global::AutoWasmApiGenerator.DefaultHttpClientHeaderHandler.Default; + _JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true }; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "{{GeneratorVersion}}")] + public void Log(string message) + => throw new global::System.NotSupportedException(); + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "{{GeneratorVersion}}")] + public async global::System.Threading.Tasks.Task LogAsync(string message) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Get; + await headerHandler.SetRequestHeaderAsync(_request_gen, global::System.Threading.CancellationToken.None); + var _url_gen = $"api/Test/Log"; + var _queries_gen = new global::System.Collections.Generic.List(); + _queries_gen.Add($"{nameof(message)}={message}"); + _url_gen = $"{_url_gen}?{string.Join("&", _queries_gen)}"; + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(); + bool.TryParse(_str_gen, out var val); + return val; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "{{GeneratorVersion}}")] + public async global::System.Threading.Tasks.Task Log2Async(string message, global::System.Threading.CancellationToken token) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Post; + await headerHandler.SetRequestHeaderAsync(_request_gen, token); + var _url_gen = $"api/Test/Log2"; + var _json_gen = global::System.Text.Json.JsonSerializer.Serialize(message); + _request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json"); + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen, token); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(token); + bool.TryParse(_str_gen, out var val); + return val; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "{{GeneratorVersion}}")] + public async global::System.Threading.Tasks.Task Log3Async(string message, string path, global::System.Threading.CancellationToken token) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Delete; + await headerHandler.SetRequestHeaderAsync(_request_gen, token); + var _url_gen = $"api/Test/Log3"; + var _queries_gen = new global::System.Collections.Generic.List(); + _queries_gen.Add($"{nameof(path)}={path}"); + _url_gen = $"{_url_gen}?{string.Join("&", _queries_gen)}"; + var _json_gen = global::System.Text.Json.JsonSerializer.Serialize(message); + _request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json"); + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen, token); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(token); + bool.TryParse(_str_gen, out var val); + return val; + } + } + } + """; + } + private const string TestSuccessCode = @"using AutoWasmApiGenerator; using System.Threading.Tasks; namespace AutoWasmApiGenerator.Test; @@ -112,111 +219,7 @@ public interface ITest } "; - private const string TestSuccessResult = - """ - // - #pragma warning disable - #nullable enable - - using Microsoft.Extensions.DependencyInjection; - - namespace AutoWasmApiGenerator.Test - { - [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] - /// - public partial class TestApiInvoker : global::AutoWasmApiGenerator.Test.ITest - { - private readonly global::System.Text.Json.JsonSerializerOptions _JSON_OPTIONS_gen; - - private readonly global::System.Net.Http.IHttpClientFactory clientFactory; - - private readonly global::AutoWasmApiGenerator.IHttpClientHeaderHandler headerHandler; - - public TestApiInvoker(global::System.Net.Http.IHttpClientFactory factory, global::System.IServiceProvider services) - { - clientFactory = factory; - headerHandler = services.GetService() ?? global::AutoWasmApiGenerator.DefaultHttpClientHeaderHandler.Default; - _JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true }; - } - - [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] - public void Log(string message) - => throw new global::System.NotSupportedException(); - - [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] - public async global::System.Threading.Tasks.Task LogAsync(string message) - { - var _client_gen = this.clientFactory.CreateClient("Test"); - var _request_gen = new global::System.Net.Http.HttpRequestMessage(); - _request_gen.Method = global::System.Net.Http.HttpMethod.Get; - await headerHandler.SetRequestHeaderAsync(_request_gen, global::System.Threading.CancellationToken.None); - var _url_gen = $"api/Test/Log"; - var _queries_gen = new global::System.Collections.Generic.List(); - _queries_gen.Add($"{nameof(message)}={message}"); - _url_gen = $"{_url_gen}?{string.Join("&", _queries_gen)}"; - _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); - var _response_gen = await _client_gen.SendAsync(_request_gen); - _response_gen.EnsureSuccessStatusCode(); - var _str_gen = await _response_gen.Content.ReadAsStringAsync(); - bool.TryParse(_str_gen, out var val); - return val; - } - - [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] - public async global::System.Threading.Tasks.Task Log2Async(string message, global::System.Threading.CancellationToken token) - { - var _client_gen = this.clientFactory.CreateClient("Test"); - var _request_gen = new global::System.Net.Http.HttpRequestMessage(); - _request_gen.Method = global::System.Net.Http.HttpMethod.Post; - await headerHandler.SetRequestHeaderAsync(_request_gen, token); - var _url_gen = $"api/Test/Log2"; - var _json_gen = global::System.Text.Json.JsonSerializer.Serialize(message); - _request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json"); - _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); - var _response_gen = await _client_gen.SendAsync(_request_gen, token); - _response_gen.EnsureSuccessStatusCode(); - var _str_gen = await _response_gen.Content.ReadAsStringAsync(token); - bool.TryParse(_str_gen, out var val); - return val; - } - - [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] - public async global::System.Threading.Tasks.Task Log3Async(string message, string path, global::System.Threading.CancellationToken token) - { - var _client_gen = this.clientFactory.CreateClient("Test"); - var _request_gen = new global::System.Net.Http.HttpRequestMessage(); - _request_gen.Method = global::System.Net.Http.HttpMethod.Delete; - await headerHandler.SetRequestHeaderAsync(_request_gen, token); - var _url_gen = $"api/Test/Log3"; - var _queries_gen = new global::System.Collections.Generic.List(); - _queries_gen.Add($"{nameof(path)}={path}"); - _url_gen = $"{_url_gen}?{string.Join("&", _queries_gen)}"; - var _json_gen = global::System.Text.Json.JsonSerializer.Serialize(message); - _request_gen.Content = new global::System.Net.Http.StringContent(_json_gen, global::System.Text.Encoding.Default, "application/json"); - _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); - var _response_gen = await _client_gen.SendAsync(_request_gen, token); - _response_gen.EnsureSuccessStatusCode(); - var _str_gen = await _response_gen.Content.ReadAsStringAsync(token); - bool.TryParse(_str_gen, out var val); - return val; - } - } - } - """; - - #endregion - - #region Wag00002 - - private const string TestWag00002 = @"using AutoWasmApiGenerator"; - private const string TestWag00002Result = @"using AutoWasmApiGenerator"; - - #endregion - - #region Wag00003 - - private const string TestWag00003 = @"using AutoWasmApiGenerator"; - private const string TestWag00003Result = @"using AutoWasmApiGenerator"; + private static string TestSuccessResult = null!; // Updated in UpdateTestString #endregion @@ -236,47 +239,52 @@ public interface ITest } """; - private const string TestFromRouteResult = - """ - // - #pragma warning disable - #nullable enable + private static string TestFromRouteResult = null!; // Updated in UpdateFromRouteTestString - using Microsoft.Extensions.DependencyInjection; + private static void UpdateFromRouteTestString() + { + TestFromRouteResult = + $$""" + // + #pragma warning disable + #nullable enable - namespace AutoWasmApiGenerator.Test - { - [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] - /// - public partial class TestApiInvoker : global::AutoWasmApiGenerator.Test.ITest + using Microsoft.Extensions.DependencyInjection; + + namespace AutoWasmApiGenerator.Test { - private readonly global::System.Text.Json.JsonSerializerOptions _JSON_OPTIONS_gen; - - private readonly global::System.Net.Http.IHttpClientFactory clientFactory; - - public TestApiInvoker(global::System.Net.Http.IHttpClientFactory factory) - { - clientFactory = factory; - _JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true }; - } - - [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] - public async global::System.Threading.Tasks.Task LogAsync(string model0) + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "{{GeneratorVersion}}")] + /// + public partial class TestApiInvoker : global::AutoWasmApiGenerator.Test.ITest { - var _client_gen = this.clientFactory.CreateClient("Test"); - var _request_gen = new global::System.Net.Http.HttpRequestMessage(); - _request_gen.Method = global::System.Net.Http.HttpMethod.Post; - var _url_gen = $"api/Test/Log/Log/{model0}"; - _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); - var _response_gen = await _client_gen.SendAsync(_request_gen); - _response_gen.EnsureSuccessStatusCode(); - var _str_gen = await _response_gen.Content.ReadAsStringAsync(); - bool.TryParse(_str_gen, out var val); - return val; + private readonly global::System.Text.Json.JsonSerializerOptions _JSON_OPTIONS_gen; + + private readonly global::System.Net.Http.IHttpClientFactory clientFactory; + + public TestApiInvoker(global::System.Net.Http.IHttpClientFactory factory) + { + clientFactory = factory; + _JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true }; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "{{GeneratorVersion}}")] + public async global::System.Threading.Tasks.Task LogAsync(string model0) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Post; + var _url_gen = $"api/Test/Log/Log/{model0}"; + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(); + bool.TryParse(_str_gen, out var val); + return val; + } } } - } - """; + """; + } #endregion @@ -322,51 +330,56 @@ public interface ITest Task Log(string message); }"; - private const string TestWag00009_ValidReturnTypeResult = - """ - // - #pragma warning disable - #nullable enable + private static string TestWag00009_ValidReturnTypeResult = null!; // Updated in UpdateWag00009TestString - using Microsoft.Extensions.DependencyInjection; + private static void UpdateWag00009TestString() + { + TestWag00009_ValidReturnTypeResult = + $$""" + // + #pragma warning disable + #nullable enable - namespace AutoWasmApiGenerator.Test - { - [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] - /// - public partial class TestApiInvoker : global::AutoWasmApiGenerator.Test.ITest + using Microsoft.Extensions.DependencyInjection; + + namespace AutoWasmApiGenerator.Test { - private readonly global::System.Text.Json.JsonSerializerOptions _JSON_OPTIONS_gen; - - private readonly global::System.Net.Http.IHttpClientFactory clientFactory; - - private readonly global::AutoWasmApiGenerator.IHttpClientHeaderHandler headerHandler; - - public TestApiInvoker(global::System.Net.Http.IHttpClientFactory factory, global::System.IServiceProvider services) - { - clientFactory = factory; - headerHandler = services.GetService() ?? global::AutoWasmApiGenerator.DefaultHttpClientHeaderHandler.Default; - _JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true }; - } - - [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "0.0.9.0")] - public async global::System.Threading.Tasks.Task Log(string message) + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "{{GeneratorVersion}}")] + /// + public partial class TestApiInvoker : global::AutoWasmApiGenerator.Test.ITest { - var _client_gen = this.clientFactory.CreateClient("Test"); - var _request_gen = new global::System.Net.Http.HttpRequestMessage(); - _request_gen.Method = global::System.Net.Http.HttpMethod.Post; - await headerHandler.SetRequestHeaderAsync(_request_gen, global::System.Threading.CancellationToken.None); - var _url_gen = $"api/Test/Log"; - _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); - var _response_gen = await _client_gen.SendAsync(_request_gen); - _response_gen.EnsureSuccessStatusCode(); - var _str_gen = await _response_gen.Content.ReadAsStringAsync(); - bool.TryParse(_str_gen, out var val); - return val; + private readonly global::System.Text.Json.JsonSerializerOptions _JSON_OPTIONS_gen; + + private readonly global::System.Net.Http.IHttpClientFactory clientFactory; + + private readonly global::AutoWasmApiGenerator.IHttpClientHeaderHandler headerHandler; + + public TestApiInvoker(global::System.Net.Http.IHttpClientFactory factory, global::System.IServiceProvider services) + { + clientFactory = factory; + headerHandler = services.GetService() ?? global::AutoWasmApiGenerator.DefaultHttpClientHeaderHandler.Default; + _JSON_OPTIONS_gen = new global::System.Text.Json.JsonSerializerOptions() { PropertyNameCaseInsensitive = true }; + } + + [global::System.CodeDom.Compiler.GeneratedCode("AutoWasmApiGenerator.HttpServiceInvokerGenerator", "{{GeneratorVersion}}")] + public async global::System.Threading.Tasks.Task Log(string message) + { + var _client_gen = this.clientFactory.CreateClient("Test"); + var _request_gen = new global::System.Net.Http.HttpRequestMessage(); + _request_gen.Method = global::System.Net.Http.HttpMethod.Post; + await headerHandler.SetRequestHeaderAsync(_request_gen, global::System.Threading.CancellationToken.None); + var _url_gen = $"api/Test/Log"; + _request_gen.RequestUri = new global::System.Uri(_url_gen, UriKind.Relative); + var _response_gen = await _client_gen.SendAsync(_request_gen); + _response_gen.EnsureSuccessStatusCode(); + var _str_gen = await _response_gen.Content.ReadAsStringAsync(); + bool.TryParse(_str_gen, out var val); + return val; + } } } - } - """; + """; + } #endregion } \ No newline at end of file diff --git a/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs b/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs index c2f8253..f82b1b0 100644 --- a/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs +++ b/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs @@ -1,3 +1,4 @@ +using System.Reflection; using FluentAssertions; using Generators.Shared; using Microsoft.CodeAnalysis; @@ -16,6 +17,7 @@ public static PortableExecutableReference[] DistinctReferences(this IEnumerable< public partial class HttpServiceInvokerGeneratorTest : IncrementalSourceGeneratorTestBase { private static readonly PortableExecutableReference[] References; + private static readonly string GeneratorVersion; static HttpServiceInvokerGeneratorTest() { @@ -24,6 +26,11 @@ static HttpServiceInvokerGeneratorTest() typeof(HttpServiceInvokerGenerator), typeof(WebControllerAttribute) }.DistinctReferences(); + GeneratorVersion = typeof(HttpServiceInvokerGenerator).Assembly.GetCustomAttribute() + ?.Version ?? throw new Exception("Unknown generator version"); + UpdateSuccessTestString(); + UpdateWag00009TestString(); + UpdateFromRouteTestString(); } [Fact] From bd0c9ba8c973dd45076e299dd97df5c1ec7d8a0e Mon Sep 17 00:00:00 2001 From: ihonliu Date: Tue, 10 Dec 2024 02:24:42 +0800 Subject: [PATCH 18/20] =?UTF-8?q?feat:=20=E5=8A=A0=E5=85=A5=E5=85=A8?= =?UTF-8?q?=E5=B1=80=E8=AE=BE=E7=BD=AE=E9=80=89=E9=A1=B9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/AutoWasmApi.Roslyn/ControllerGenerator.cs | 13 +++- .../HttpServiceInvokerGenerator.cs | 15 +++- .../Options/GlobalOptions.cs | 49 ++++++++++++ .../build/AutoWasmApiGenerator.props | 14 ++++ .../AutoWasmApiGenerator.Test.csproj | 16 ++-- .../GlobalOptionsTest.cs | 76 +++++++++++++++++++ 6 files changed, 173 insertions(+), 10 deletions(-) create mode 100644 src/AutoWasmApi.Roslyn/Options/GlobalOptions.cs create mode 100644 src/AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props create mode 100644 src/AutoWasmApiGenerator.Test/GlobalOptionsTest.cs diff --git a/src/AutoWasmApi.Roslyn/ControllerGenerator.cs b/src/AutoWasmApi.Roslyn/ControllerGenerator.cs index 6e529fc..7e07580 100644 --- a/src/AutoWasmApi.Roslyn/ControllerGenerator.cs +++ b/src/AutoWasmApi.Roslyn/ControllerGenerator.cs @@ -5,6 +5,7 @@ using System.Linq; using System.Text.RegularExpressions; using System.Threading; +using AutoWasmApiGenerator.Options; using Generators.Shared; using Generators.Shared.Builder; using Microsoft.CodeAnalysis; @@ -25,12 +26,18 @@ public void Initialize(IncrementalGeneratorInitializationContext context) // WebControllerAttributeFullName // , static (node, _) => true // , static (ctx, _) => ctx); + var globalOptions = context.AnalyzerConfigOptionsProvider.Select(GlobalOptions.Select); - context.RegisterSourceOutput(context.CompilationProvider, static (context, compilation) => + context.RegisterSourceOutput(context.CompilationProvider.Combine(globalOptions), static (context, compilationData) => { //try //{ - if (!compilation.Assembly.HasAttribute(WebControllerAssemblyAttributeFullName)) + var compilation = compilationData.Left; + var globalOptions = compilationData.Right; + var generate = compilation.Assembly.HasAttribute(WebControllerAssemblyAttributeFullName) + || globalOptions.GenerateWebController; + + if (!generate) { return; } @@ -112,7 +119,7 @@ private static MethodBuilder BuildMethod((IMethodSymbol, AttributeData?) data, s { methodRoute = methodScoped; } - else if(Regex.Match(customRoute, "{.+}").Success) + else if (Regex.Match(customRoute, "{.+}").Success) { methodRoute = $"{methodScoped}/{customRoute}"; } diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs index 23129f3..333f327 100644 --- a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs +++ b/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs @@ -1,5 +1,7 @@ using System; +using System.Diagnostics; using System.Linq; +using AutoWasmApiGenerator.Options; using Generators.Shared; using Microsoft.CodeAnalysis; using static AutoWasmApiGenerator.GeneratorHelpers; @@ -13,11 +15,20 @@ public class HttpServiceInvokerGenerator : IIncrementalGenerator public void Initialize(IncrementalGeneratorInitializationContext context) { - context.RegisterSourceOutput(context.CompilationProvider, static (context, compilation) => + // if (!Debugger.IsAttached) + // { + // Debugger.Launch(); + // } + var globalOptions = context.AnalyzerConfigOptionsProvider.Select(GlobalOptions.Select); + context.RegisterSourceOutput(context.CompilationProvider.Combine(globalOptions), static (context, compilationData) => { try { - if (!compilation.Assembly.HasAttribute(ApiInvokerAssemblyAttributeFullName)) + var compilation = compilationData.Left; + var globalOptions = compilationData.Right; + var generate = compilation.Assembly.HasAttribute(ApiInvokerAssemblyAttributeFullName) + || globalOptions.GenerateInvoker; + if (!generate) { return; } diff --git a/src/AutoWasmApi.Roslyn/Options/GlobalOptions.cs b/src/AutoWasmApi.Roslyn/Options/GlobalOptions.cs new file mode 100644 index 0000000..0b3e2d7 --- /dev/null +++ b/src/AutoWasmApi.Roslyn/Options/GlobalOptions.cs @@ -0,0 +1,49 @@ +using System; +using System.Threading; +using Microsoft.CodeAnalysis.Diagnostics; + +namespace AutoWasmApiGenerator.Options; + +public sealed class GlobalOptions +{ + public GlobalOptions(AnalyzerConfigOptions options) + { + if (options.TryGetValue("build_property.MSBuildProjectFullPath", out var projectFullPath)) + { + ProjectFullPath = projectFullPath; + } + + ProjectFullPath ??= string.Empty; + + GenerateInvoker = + options.TryGetValue("build_property.AutoWasmApiGenerator_GenerateInvoker", out var generateInvoker) && + generateInvoker is { Length: > 0 } && + generateInvoker.Equals("true", StringComparison.OrdinalIgnoreCase); + + GenerateWebController = + options.TryGetValue("build_property.AutoWasmApiGenerator_GenerateWebController", out var generateWebController) && + generateWebController is { Length: > 0 } && + generateWebController.Equals("true", StringComparison.OrdinalIgnoreCase); + } + + /// + /// + /// + public bool GenerateInvoker { get; } + + /// + /// + /// + public bool GenerateWebController { get; } + + /// + /// + /// + public string? ProjectFullPath { get; } + + public static GlobalOptions Select(AnalyzerConfigOptionsProvider provider, CancellationToken token) + { + token.ThrowIfCancellationRequested(); + return new GlobalOptions(provider.GlobalOptions); + } +} \ No newline at end of file diff --git a/src/AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props b/src/AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props new file mode 100644 index 0000000..eda792e --- /dev/null +++ b/src/AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + diff --git a/src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj b/src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj index 8f8c904..8976994 100644 --- a/src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj +++ b/src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj @@ -7,8 +7,18 @@ false true + true + true + + + + + + + + @@ -28,7 +38,7 @@ - + @@ -46,10 +56,6 @@ - - - - diff --git a/src/AutoWasmApiGenerator.Test/GlobalOptionsTest.cs b/src/AutoWasmApiGenerator.Test/GlobalOptionsTest.cs new file mode 100644 index 0000000..a9d518a --- /dev/null +++ b/src/AutoWasmApiGenerator.Test/GlobalOptionsTest.cs @@ -0,0 +1,76 @@ +using System.Diagnostics.CodeAnalysis; +using AutoWasmApiGenerator.Options; +using FluentAssertions; +using Microsoft.CodeAnalysis; +using Microsoft.CodeAnalysis.Diagnostics; + +namespace AutoWasmApiGenerator.Test; + +public class GlobalOptionsTest +{ + private static readonly GlobalOptions LocalGlobalOptionsForTests = CreateOptions(new AnalyzerConfigOptionsDummy()); + + private static GlobalOptions CreateOptions(AnalyzerConfigOptionsDummy dummy) + { + return GlobalOptions.Select(new AnalyzerConfigOptionsProviderDummy(dummy), default); + } + + [Fact] + public void GlobalDefault() + { + var globalOptions = LocalGlobalOptionsForTests; + globalOptions.ProjectFullPath.Should().BeEmpty(); + globalOptions.GenerateInvoker.Should().BeFalse(); + globalOptions.GenerateWebController.Should().BeFalse(); + } + + [Fact] + public void GlobalSettings_CanReadAll() + { + var globalOptions = CreateOptions(new AnalyzerConfigOptionsDummy() + { + MSBuildProjectFullPath = "projectFullPath.csproj", + AutoWasmApiGenerator_GenerateInvoker = "true", + AutoWasmApiGenerator_GenerateWebController = "true" + }); + globalOptions.ProjectFullPath.Should().Be("projectFullPath.csproj"); + globalOptions.GenerateInvoker.Should().BeTrue(); + globalOptions.GenerateWebController.Should().BeTrue(); + } +} + +public class AnalyzerConfigOptionsDummy : AnalyzerConfigOptions +{ + // ReSharper disable InconsistentNaming + public string? MSBuildProjectFullPath { get; init; } + public string? AutoWasmApiGenerator_GenerateInvoker { get; init; } + public string? AutoWasmApiGenerator_GenerateWebController { get; init; } + // ReSharper restore InconsistentNaming + + public override bool TryGetValue(string key, [NotNullWhen(true)] out string? value) + { + value = key switch + { + "build_property.MSBuildProjectFullPath" => MSBuildProjectFullPath, + "build_property.AutoWasmApiGenerator_GenerateInvoker" => AutoWasmApiGenerator_GenerateInvoker, + "build_property.AutoWasmApiGenerator_GenerateWebController" => AutoWasmApiGenerator_GenerateWebController, + _ => null + }; + return value != null; + } +} + +public sealed class AnalyzerConfigOptionsProviderDummy(AnalyzerConfigOptions globalOptions) : AnalyzerConfigOptionsProvider +{ + public override AnalyzerConfigOptions GetOptions(SyntaxTree tree) + { + throw new NotImplementedException(); + } + + public override AnalyzerConfigOptions GetOptions(AdditionalText textFile) + { + return GlobalOptions; + } + + public override AnalyzerConfigOptions GlobalOptions { get; } = globalOptions; +} \ No newline at end of file From 47a51a93e3914e5cb08ba283f117028120b405c6 Mon Sep 17 00:00:00 2001 From: ihonliu Date: Tue, 10 Dec 2024 15:33:31 +0800 Subject: [PATCH 19/20] =?UTF-8?q?refactor:=20=E4=BF=AE=E6=94=B9AutoWasmApi?= =?UTF-8?q?=E9=A1=B9=E7=9B=AE=E4=BD=8D=E7=BD=AE=EF=BC=8C=E7=94=A8=E4=BA=8E?= =?UTF-8?q?=E9=A1=B9=E7=9B=AE=E7=BB=9F=E4=B8=80=E8=AE=BE=E7=BD=AE=E7=9B=B8?= =?UTF-8?q?=E5=85=B3=E5=B1=9E=E6=80=A7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md | 0 .../AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md | 0 .../AutoWasmApi.Roslyn/Attributes/NullableAttributes.cs | 0 .../AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj | 4 ++-- .../AutoWasmApi.Roslyn/ControllerGenerator.cs | 0 .../AutoWasmApi.Roslyn/DiagnosticDefinitions.cs | 0 .../Extensions/MethodCheckerExtensions.cs | 0 .../AutoWasmApi.Roslyn/GeneratorHelpers.cs | 0 .../AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs | 0 .../HttpServiceInvokerGeneratorImpl.cs | 0 .../AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs | 0 .../AutoWasmApi.Roslyn/Options/GlobalOptions.cs | 0 .../AutoWasmApi.Roslyn/Properties/launchSettings.json | 0 .../AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props | 0 .../AutoWasmApi.Roslyn/usings.cs | 0 .../AutoWasmApiGenerator.Test.csproj | 2 +- .../AutoWasmApiGenerator.Test/GlobalOptionsTest.cs | 0 .../HttpServiceInvokerGeneratorTest.Constants.cs | 0 .../HttpServiceInvokerGeneratorTest.cs | 0 .../IncrementalSourceGeneratorTestBase.cs | 0 .../Attributes/ApiInvokeNotSupportedAttribute.cs | 0 .../Attributes/ApiInvokerAssemblyAttribute.cs | 0 .../Attributes/ApiInvokerGenerateAttribute.cs | 0 .../Attributes/NullableAttributes.cs | 0 .../Attributes/WebControllerAssemblyAttribute.cs | 0 .../Attributes/WebControllerAttribute.cs | 0 .../Attributes/WebMethodAttribute.cs | 0 .../Attributes/WebMethodParameterBindingAttribute.cs | 0 .../AutoWasmApiGenerator.csproj | 6 ------ .../IHttpClientHeaderHandler.cs | 0 .../{ => AutoWasmApiGenerator}/Models/BindingType.cs | 0 .../{ => AutoWasmApiGenerator}/Models/IsExternalInit.cs | 0 .../{ => AutoWasmApiGenerator}/Models/WebMethod.cs | 0 .../{ => AutoWasmApiGenerator}/readme.md | 0 src/AutoWasmApiGenerator/Directory.Build.props | 9 +++++++++ .../Blazor.Test.Client/Blazor.Test.Client.csproj | 4 ++-- src/Blazor.Test/Blazor.Test/Blazor.Test.csproj | 4 ++-- src/Blazor.Test/Blazor.Test/Class1.cs | 2 +- src/InjectTest/InjectTest.csproj | 2 +- src/MT.Generators.sln | 6 +++--- 40 files changed, 21 insertions(+), 18 deletions(-) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/Attributes/NullableAttributes.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj (60%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/ControllerGenerator.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/Extensions/MethodCheckerExtensions.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/GeneratorHelpers.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/Options/GlobalOptions.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/Properties/launchSettings.json (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApi.Roslyn/usings.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj (96%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApiGenerator.Test/GlobalOptionsTest.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs (100%) rename src/{ => AutoWasmApiGenerator}/AutoWasmApiGenerator.Test/IncrementalSourceGeneratorTestBase.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Attributes/ApiInvokeNotSupportedAttribute.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Attributes/ApiInvokerAssemblyAttribute.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Attributes/ApiInvokerGenerateAttribute.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Attributes/NullableAttributes.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Attributes/WebControllerAssemblyAttribute.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Attributes/WebControllerAttribute.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Attributes/WebMethodAttribute.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Attributes/WebMethodParameterBindingAttribute.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/AutoWasmApiGenerator.csproj (78%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/IHttpClientHeaderHandler.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Models/BindingType.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Models/IsExternalInit.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/Models/WebMethod.cs (100%) rename src/AutoWasmApiGenerator/{ => AutoWasmApiGenerator}/readme.md (100%) create mode 100644 src/AutoWasmApiGenerator/Directory.Build.props diff --git a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md similarity index 100% rename from src/AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AnalyzerReleases.Shipped.md diff --git a/src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md similarity index 100% rename from src/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AnalyzerReleases.Unshipped.md diff --git a/src/AutoWasmApi.Roslyn/Attributes/NullableAttributes.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/Attributes/NullableAttributes.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/Attributes/NullableAttributes.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/Attributes/NullableAttributes.cs diff --git a/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj similarity index 60% rename from src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj index 4ad31fb..b3104d3 100644 --- a/src/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj +++ b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj @@ -7,7 +7,7 @@ 0.1.1 - - + + diff --git a/src/AutoWasmApi.Roslyn/ControllerGenerator.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/ControllerGenerator.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/ControllerGenerator.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/ControllerGenerator.cs diff --git a/src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/DiagnosticDefinitions.cs diff --git a/src/AutoWasmApi.Roslyn/Extensions/MethodCheckerExtensions.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/Extensions/MethodCheckerExtensions.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/Extensions/MethodCheckerExtensions.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/Extensions/MethodCheckerExtensions.cs diff --git a/src/AutoWasmApi.Roslyn/GeneratorHelpers.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/GeneratorHelpers.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/GeneratorHelpers.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/GeneratorHelpers.cs diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/HttpServiceInvokerGenerator.cs diff --git a/src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/HttpServiceInvokerGeneratorImpl.cs diff --git a/src/AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/IHttpServiceInvokerGenerator.cs diff --git a/src/AutoWasmApi.Roslyn/Options/GlobalOptions.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/Options/GlobalOptions.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/Options/GlobalOptions.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/Options/GlobalOptions.cs diff --git a/src/AutoWasmApi.Roslyn/Properties/launchSettings.json b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/Properties/launchSettings.json similarity index 100% rename from src/AutoWasmApi.Roslyn/Properties/launchSettings.json rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/Properties/launchSettings.json diff --git a/src/AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props similarity index 100% rename from src/AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/build/AutoWasmApiGenerator.props diff --git a/src/AutoWasmApi.Roslyn/usings.cs b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/usings.cs similarity index 100% rename from src/AutoWasmApi.Roslyn/usings.cs rename to src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/usings.cs diff --git a/src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj similarity index 96% rename from src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj index 8976994..fc4f8d3 100644 --- a/src/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj +++ b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj @@ -56,6 +56,6 @@ - + diff --git a/src/AutoWasmApiGenerator.Test/GlobalOptionsTest.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/GlobalOptionsTest.cs similarity index 100% rename from src/AutoWasmApiGenerator.Test/GlobalOptionsTest.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/GlobalOptionsTest.cs diff --git a/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs similarity index 100% rename from src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.Constants.cs diff --git a/src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs similarity index 100% rename from src/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/HttpServiceInvokerGeneratorTest.cs diff --git a/src/AutoWasmApiGenerator.Test/IncrementalSourceGeneratorTestBase.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/IncrementalSourceGeneratorTestBase.cs similarity index 100% rename from src/AutoWasmApiGenerator.Test/IncrementalSourceGeneratorTestBase.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/IncrementalSourceGeneratorTestBase.cs diff --git a/src/AutoWasmApiGenerator/Attributes/ApiInvokeNotSupportedAttribute.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/ApiInvokeNotSupportedAttribute.cs similarity index 100% rename from src/AutoWasmApiGenerator/Attributes/ApiInvokeNotSupportedAttribute.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/ApiInvokeNotSupportedAttribute.cs diff --git a/src/AutoWasmApiGenerator/Attributes/ApiInvokerAssemblyAttribute.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/ApiInvokerAssemblyAttribute.cs similarity index 100% rename from src/AutoWasmApiGenerator/Attributes/ApiInvokerAssemblyAttribute.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/ApiInvokerAssemblyAttribute.cs diff --git a/src/AutoWasmApiGenerator/Attributes/ApiInvokerGenerateAttribute.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/ApiInvokerGenerateAttribute.cs similarity index 100% rename from src/AutoWasmApiGenerator/Attributes/ApiInvokerGenerateAttribute.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/ApiInvokerGenerateAttribute.cs diff --git a/src/AutoWasmApiGenerator/Attributes/NullableAttributes.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/NullableAttributes.cs similarity index 100% rename from src/AutoWasmApiGenerator/Attributes/NullableAttributes.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/NullableAttributes.cs diff --git a/src/AutoWasmApiGenerator/Attributes/WebControllerAssemblyAttribute.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/WebControllerAssemblyAttribute.cs similarity index 100% rename from src/AutoWasmApiGenerator/Attributes/WebControllerAssemblyAttribute.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/WebControllerAssemblyAttribute.cs diff --git a/src/AutoWasmApiGenerator/Attributes/WebControllerAttribute.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/WebControllerAttribute.cs similarity index 100% rename from src/AutoWasmApiGenerator/Attributes/WebControllerAttribute.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/WebControllerAttribute.cs diff --git a/src/AutoWasmApiGenerator/Attributes/WebMethodAttribute.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/WebMethodAttribute.cs similarity index 100% rename from src/AutoWasmApiGenerator/Attributes/WebMethodAttribute.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/WebMethodAttribute.cs diff --git a/src/AutoWasmApiGenerator/Attributes/WebMethodParameterBindingAttribute.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/WebMethodParameterBindingAttribute.cs similarity index 100% rename from src/AutoWasmApiGenerator/Attributes/WebMethodParameterBindingAttribute.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Attributes/WebMethodParameterBindingAttribute.cs diff --git a/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj similarity index 78% rename from src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj index e7d6faf..307dc6e 100644 --- a/src/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj +++ b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj @@ -2,14 +2,8 @@ netstandard2.0;net6.0;net8.0 - latest - enable - true - Generated true True - MarvelTiter - 0.1.1 MIT readme.md True diff --git a/src/AutoWasmApiGenerator/IHttpClientHeaderHandler.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/IHttpClientHeaderHandler.cs similarity index 100% rename from src/AutoWasmApiGenerator/IHttpClientHeaderHandler.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/IHttpClientHeaderHandler.cs diff --git a/src/AutoWasmApiGenerator/Models/BindingType.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Models/BindingType.cs similarity index 100% rename from src/AutoWasmApiGenerator/Models/BindingType.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Models/BindingType.cs diff --git a/src/AutoWasmApiGenerator/Models/IsExternalInit.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Models/IsExternalInit.cs similarity index 100% rename from src/AutoWasmApiGenerator/Models/IsExternalInit.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Models/IsExternalInit.cs diff --git a/src/AutoWasmApiGenerator/Models/WebMethod.cs b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/Models/WebMethod.cs similarity index 100% rename from src/AutoWasmApiGenerator/Models/WebMethod.cs rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/Models/WebMethod.cs diff --git a/src/AutoWasmApiGenerator/readme.md b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/readme.md similarity index 100% rename from src/AutoWasmApiGenerator/readme.md rename to src/AutoWasmApiGenerator/AutoWasmApiGenerator/readme.md diff --git a/src/AutoWasmApiGenerator/Directory.Build.props b/src/AutoWasmApiGenerator/Directory.Build.props new file mode 100644 index 0000000..1c411d5 --- /dev/null +++ b/src/AutoWasmApiGenerator/Directory.Build.props @@ -0,0 +1,9 @@ + + + + 0.1.1 + latest + enable + MarvelTiter + + diff --git a/src/Blazor.Test/Blazor.Test.Client/Blazor.Test.Client.csproj b/src/Blazor.Test/Blazor.Test.Client/Blazor.Test.Client.csproj index b476d21..093d759 100644 --- a/src/Blazor.Test/Blazor.Test.Client/Blazor.Test.Client.csproj +++ b/src/Blazor.Test/Blazor.Test.Client/Blazor.Test.Client.csproj @@ -19,8 +19,8 @@ - - + + diff --git a/src/Blazor.Test/Blazor.Test/Blazor.Test.csproj b/src/Blazor.Test/Blazor.Test/Blazor.Test.csproj index acef5ec..88e5920 100644 --- a/src/Blazor.Test/Blazor.Test/Blazor.Test.csproj +++ b/src/Blazor.Test/Blazor.Test/Blazor.Test.csproj @@ -8,8 +8,8 @@ - - + + diff --git a/src/Blazor.Test/Blazor.Test/Class1.cs b/src/Blazor.Test/Blazor.Test/Class1.cs index 74e5609..fc807f7 100644 --- a/src/Blazor.Test/Blazor.Test/Class1.cs +++ b/src/Blazor.Test/Blazor.Test/Class1.cs @@ -49,7 +49,7 @@ public Task Log2Async(string message, CancellationToken token) public Task Log3Async(string message, string path, CancellationToken token) { throw new NotImplementedException(); - return Task.FromResult(message.Length > 5); + // return Task.FromResult(message.Length > 5); } } diff --git a/src/InjectTest/InjectTest.csproj b/src/InjectTest/InjectTest.csproj index dd31cb2..836cdc4 100644 --- a/src/InjectTest/InjectTest.csproj +++ b/src/InjectTest/InjectTest.csproj @@ -7,8 +7,8 @@ - + diff --git a/src/MT.Generators.sln b/src/MT.Generators.sln index 3a1d0a7..b7fc716 100644 --- a/src/MT.Generators.sln +++ b/src/MT.Generators.sln @@ -5,7 +5,7 @@ VisualStudioVersion = 17.11.35111.106 MinimumVisualStudioVersion = 10.0.40219.1 Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "TestProject1", "TestProject1\TestProject1.csproj", "{6994C35E-E92F-418C-B5C9-B437FF25A6A0}" EndProject -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AutoWasmApiGenerator", "AutoWasmApiGenerator\AutoWasmApiGenerator.csproj", "{9C1A1324-0111-468F-9644-F80F72A66DC5}" +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AutoWasmApiGenerator", "AutoWasmApiGenerator\AutoWasmApiGenerator\AutoWasmApiGenerator.csproj", "{9C1A1324-0111-468F-9644-F80F72A66DC5}" EndProject Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AutoInjectGenerator", "AutoInjectGenerator\AutoInjectGenerator.csproj", "{1CA123D0-5B2A-4BBD-A90E-B58E01A7B2CA}" EndProject @@ -41,9 +41,9 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Shared", "Shared", "{71D54F EndProject Project("{D954291E-2A0B-460D-934E-DC6B0785DB48}") = "Generators.Shared", "Shared\Generators.Shared\Generators.Shared.shproj", "{0C3325BB-C186-4E45-A567-69DAF5411E8B}" EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AutoWasmApiGenerator.Test", "AutoWasmApiGenerator.Test\AutoWasmApiGenerator.Test.csproj", "{3ED9CDA7-B256-4B4C-85DF-E1EA62715428}" +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AutoWasmApiGenerator.Test", "AutoWasmApiGenerator\AutoWasmApiGenerator.Test\AutoWasmApiGenerator.Test.csproj", "{3ED9CDA7-B256-4B4C-85DF-E1EA62715428}" EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AutoWasmApi.Roslyn", "AutoWasmApi.Roslyn\AutoWasmApi.Roslyn.csproj", "{144DCF75-901E-40EE-8449-FC404D57017D}" +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AutoWasmApi.Roslyn", "AutoWasmApiGenerator\AutoWasmApi.Roslyn\AutoWasmApi.Roslyn.csproj", "{144DCF75-901E-40EE-8449-FC404D57017D}" EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution From a2b73dd01b69b4eef902e3775e4c6fdc7757cb98 Mon Sep 17 00:00:00 2001 From: ihonliu Date: Wed, 11 Dec 2024 00:22:07 +0800 Subject: [PATCH 20/20] =?UTF-8?q?bug:=20=E4=BF=AE=E5=A4=8D=E8=87=AA?= =?UTF-8?q?=E5=AE=9A=E4=B9=89=E5=B1=9E=E6=80=A7=E6=97=A0=E6=95=88=E7=9A=84?= =?UTF-8?q?=E9=97=AE=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj | 13 +++++-------- .../AutoWasmApiGenerator.Test.csproj | 6 ------ .../AutoWasmApiGenerator.csproj | 5 +++++ src/AutoWasmApiGenerator/Directory.Build.props | 1 + src/MT.Generators.sln | 3 +++ 5 files changed, 14 insertions(+), 14 deletions(-) diff --git a/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj index b3104d3..f8f722b 100644 --- a/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj +++ b/src/AutoWasmApiGenerator/AutoWasmApi.Roslyn/AutoWasmApi.Roslyn.csproj @@ -1,13 +1,10 @@  - - netstandard2.0 - AutoWasmApiGenerator - latest - 0.1.1 - + + netstandard2.0 + - - + + diff --git a/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj index fc4f8d3..0a68c0a 100644 --- a/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj +++ b/src/AutoWasmApiGenerator/AutoWasmApiGenerator.Test/AutoWasmApiGenerator.Test.csproj @@ -13,12 +13,6 @@ - - - - - - diff --git a/src/AutoWasmApiGenerator/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj index 307dc6e..5d1a101 100644 --- a/src/AutoWasmApiGenerator/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj +++ b/src/AutoWasmApiGenerator/AutoWasmApiGenerator/AutoWasmApiGenerator.csproj @@ -16,6 +16,11 @@ + + + true + build + diff --git a/src/AutoWasmApiGenerator/Directory.Build.props b/src/AutoWasmApiGenerator/Directory.Build.props index 1c411d5..699419b 100644 --- a/src/AutoWasmApiGenerator/Directory.Build.props +++ b/src/AutoWasmApiGenerator/Directory.Build.props @@ -5,5 +5,6 @@ latest enable MarvelTiter + AutoWasmApiGenerator diff --git a/src/MT.Generators.sln b/src/MT.Generators.sln index b7fc716..91a50cb 100644 --- a/src/MT.Generators.sln +++ b/src/MT.Generators.sln @@ -16,6 +16,9 @@ EndProject Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AutoInject.Roslyn", "AutoInject.Roslyn\AutoInject.Roslyn.csproj", "{E48C1C4F-8111-4513-86EC-0C968E5437CD}" EndProject Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "AutoWasm", "AutoWasm", "{0D1AD616-22D2-4D10-BD23-92040AFD90D5}" + ProjectSection(SolutionItems) = preProject + AutoWasmApiGenerator\Directory.Build.props = AutoWasmApiGenerator\Directory.Build.props + EndProjectSection EndProject Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Blazor.Test", "Blazor.Test\Blazor.Test\Blazor.Test.csproj", "{4A47713C-2332-40C1-9AAB-7B2A14D148A2}" EndProject