mirror of
https://github.com/ckaczor/sqltoolsservice.git
synced 2026-01-21 17:24:03 -05:00
156 lines
8.4 KiB
C#
156 lines
8.4 KiB
C#
//
|
|
// Copyright (c) Microsoft. All rights reserved.
|
|
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
|
|
//
|
|
|
|
using Microsoft.CodeAnalysis.CSharp;
|
|
using Microsoft.CodeAnalysis.CSharp.Syntax;
|
|
using Microsoft.SqlTools.ServiceLayer.AzureFunctions.Contracts;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
|
|
namespace Microsoft.SqlTools.ServiceLayer.AzureFunctions
|
|
{
|
|
internal static class AzureFunctionsUtils
|
|
{
|
|
private const string FUNCTION_NAME_ATTRIBUTE_NAME = "FunctionName";
|
|
private const string NET5_FUNCTION_ATTRIBUTE_NAME = "Function";
|
|
private const string HTTP_TRIGGER_ATTRIBUTE_NAME = "HttpTrigger";
|
|
private const string ROUTE_ARGUMENT_NAME = "Route";
|
|
|
|
/// <summary>
|
|
/// Gets all the methods in the syntax tree with an Azure Function attribute
|
|
/// </summary>
|
|
public static IEnumerable<MethodDeclarationSyntax> GetMethodsWithFunctionAttributes(CompilationUnitSyntax root)
|
|
{
|
|
// Look for Azure Functions in the file
|
|
// Get all method declarations
|
|
IEnumerable<MethodDeclarationSyntax> methodDeclarations = root.DescendantNodes().OfType<MethodDeclarationSyntax>();
|
|
|
|
// .NET 5 is not currently supported for sql bindings, so an error should be returned if this file has .NET 5 style Azure Functions
|
|
if (HasNet5StyleAzureFunctions(methodDeclarations))
|
|
{
|
|
throw new Exception(SR.SqlBindingsNet5NotSupported);
|
|
}
|
|
|
|
// get all the method declarations with the FunctionName attribute
|
|
IEnumerable<MethodDeclarationSyntax> methodsWithFunctionAttributes = methodDeclarations.Where(md => md.AttributeLists.Where(a => a.Attributes.Where(attr => attr.Name.ToString().Equals(FUNCTION_NAME_ATTRIBUTE_NAME)).Any()).Any());
|
|
|
|
return methodsWithFunctionAttributes;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the info for a HttpTriggerBinding for the specified method (if such a binding exists)
|
|
/// </summary>
|
|
/// <param name="m">The method</param>
|
|
/// <returns>The HttpTriggerBinding, or null if none exists</returns>
|
|
public static HttpTriggerBinding? GetHttpTriggerBinding(this MethodDeclarationSyntax m)
|
|
{
|
|
var httpTriggerAttribute = m.GetHttpTriggerAttribute();
|
|
return httpTriggerAttribute == null ? null : new HttpTriggerBinding(httpTriggerAttribute.GetHttpRoute(), httpTriggerAttribute.GetHttpOperations());
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the HttpTrigger attribute on the parameters for this method if one exists
|
|
/// https://docs.microsoft.com/en-us/azure/azure-functions/functions-bindings-http-webhook-trigger
|
|
/// </summary>
|
|
/// <param name="m">The method</param>
|
|
/// <returns>The attribute if such a binding exists</returns>
|
|
public static AttributeSyntax? GetHttpTriggerAttribute(this MethodDeclarationSyntax m)
|
|
{
|
|
return m
|
|
.ParameterList
|
|
.Parameters // Get all the parameters for the method
|
|
.SelectMany(p =>
|
|
p.AttributeLists // Get a list of all attributes on any of the parameters
|
|
.SelectMany(al => al.Attributes)
|
|
).Where(a => a.Name.ToString().Equals(HTTP_TRIGGER_ATTRIBUTE_NAME) // Find any HttpTrigger attributes
|
|
).FirstOrDefault(); // Get the first one available - there should only ever be 0 or 1
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the route from the HttpTrigger attribute
|
|
/// https://docs.microsoft.com/en-us/azure/azure-functions/functions-bindings-http-webhook-trigger?tabs=in-process%2Cfunctionsv2&pivots=programming-language-csharp#attributes
|
|
/// </summary>
|
|
/// <param name="m">The attribute</param>
|
|
/// <returns>The name of the route, or null if no route is specified (or there isn't an HttpTrigger binding)</returns>
|
|
public static string? GetHttpRoute(this AttributeSyntax a)
|
|
{
|
|
return a.ArgumentList
|
|
?.Arguments
|
|
.Where(a => a.ChildNodes().OfType<NameEqualsSyntax>().Where(nes => nes.Name.ToString().Equals(ROUTE_ARGUMENT_NAME)).Any()) // Find the Route argument - it should always be a named argument
|
|
.FirstOrDefault()
|
|
?.ChildNodes()
|
|
.OfType<ExpressionSyntax>() // Find the child identifier node with our value
|
|
.Where(le => le.Kind() != SyntaxKind.NullLiteralExpression) // Skip the null expressions so they aren't ToString()'d into "null"
|
|
.FirstOrDefault()
|
|
?.ToString()
|
|
.TrimStringQuotes();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the operations (methods) on an HttpTrigger attribute. These are string params arguments to the attribute
|
|
/// https://docs.microsoft.com/en-us/azure/azure-functions/functions-bindings-http-webhook-trigger?tabs=in-process%2Cfunctionsv2&pivots=programming-language-csharp#attributes
|
|
/// </summary>
|
|
/// <param name="a">The attribute</param>
|
|
/// <returns>The operations (methods) specified</returns>
|
|
public static string[]? GetHttpOperations(this AttributeSyntax a)
|
|
{
|
|
return a.ArgumentList
|
|
?.Arguments
|
|
.Where(a => a.Expression.Kind() == SyntaxKind.StringLiteralExpression && a.NameEquals == null) // Operations are string literals who don't have a name (Route is always a named param)
|
|
.Select(a => a.ToString().TrimStringQuotes().ToUpper()) // upper case for consistent naming
|
|
.ToArray();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the function name from the FunctionName attribute on a method
|
|
/// </summary>
|
|
/// <param name="m">The method</param>
|
|
/// <returns>The function name, or an empty string if the name attribute doesn't exist</returns>
|
|
public static string GetFunctionName(this MethodDeclarationSyntax m)
|
|
{
|
|
// Note that we return an empty string as the default because a null name isn't valid - every function
|
|
// should have a name. So we should never actually hit that scenario, but just to be safe we return the
|
|
// empty string as the default in case we hit some unexpected edge case.
|
|
return m
|
|
.AttributeLists // Get all the attribute lists on the method
|
|
.Select(a =>
|
|
a.Attributes.Where(
|
|
attr => attr.Name.ToString().Equals(AzureFunctionsUtils.FUNCTION_NAME_ATTRIBUTE_NAME) // Find any FunctionName attributes
|
|
).FirstOrDefault() // Get the first one available - there should only ever be 0 or 1
|
|
).Where(a => // Filter out any that didn't have a FunctionName attribute
|
|
a != null
|
|
).FirstOrDefault() // Get the first one available - there should only ever be 0 or 1
|
|
?.ArgumentList // Get all the arguments for the attribute
|
|
?.Arguments
|
|
.FirstOrDefault() // The first argument is the function name
|
|
?.ToString()
|
|
.TrimStringQuotes() ?? "";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Removes the quotes (and $ for interpolated strings) around a string literal
|
|
/// </summary>
|
|
/// <param name="s">The string</param>
|
|
/// <returns>The string without quotes</returns>
|
|
public static string TrimStringQuotes(this string s)
|
|
{
|
|
return s
|
|
.TrimStart('$') // Remove $ from interpolated string, since this will always be outside the quotes we can just trim
|
|
.Trim('\"'); // Trim off the quotes from the string value - additional quotes at the beginning and end aren't valid syntax so it's fine to trim
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks if any of the method declarations have .NET 5 style Azure Function attributes
|
|
/// .NET 5 AFs use the Function attribute, while .NET 3.1 AFs use FunctionName attribute
|
|
/// </summary>
|
|
public static bool HasNet5StyleAzureFunctions(IEnumerable<MethodDeclarationSyntax> methodDeclarations)
|
|
{
|
|
// get all the method declarations with the Function attribute
|
|
return methodDeclarations.Any(md => md.AttributeLists.Any(al => al.Attributes.Any(attr => attr.Name.ToString().Equals(NET5_FUNCTION_ATTRIBUTE_NAME))));
|
|
}
|
|
}
|
|
}
|