Files
2018-10-10 14:40:52 +01:00

103 lines
3.5 KiB
C#

using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using Semmle.Extraction.Entities;
using System.Collections.Generic;
using System.Linq;
namespace Semmle.Extraction.CSharp.Entities
{
public abstract class CachedSymbol<T> : CachedEntity<T> where T : ISymbol
{
public CachedSymbol(Context cx, T init)
: base(cx, init) { }
public virtual Type ContainingType => symbol.ContainingType != null ? Type.Create(Context, symbol.ContainingType) : null;
public void ExtractModifiers()
{
Modifier.ExtractModifiers(Context, this, symbol);
}
protected void ExtractAttributes()
{
// Only extract attributes for source declarations
if (ReferenceEquals(symbol, symbol.OriginalDefinition))
Attribute.ExtractAttributes(Context, symbol, this);
}
protected void ExtractCompilerGenerated()
{
if (symbol.IsImplicitlyDeclared)
Context.Emit(Tuples.compiler_generated(this));
}
/// <summary>
/// The location which is stored in the database and is used when highlighing source code.
/// It's generally short, e.g. a method name.
/// </summary>
public override Microsoft.CodeAnalysis.Location ReportingLocation => symbol.Locations.FirstOrDefault();
/// <summary>
/// The full text span of the entity, e.g. for binding comments.
/// </summary>
public virtual Microsoft.CodeAnalysis.Location FullLocation => symbol.Locations.FirstOrDefault();
public virtual IEnumerable<Extraction.Entities.Location> Locations
{
get
{
var loc = ReportingLocation;
if (loc != null)
{
// Some built in operators lack locations, so loc is null.
yield return Context.Create(ReportingLocation);
if (Context.Extractor.OutputPath != null && loc.Kind == LocationKind.SourceFile)
yield return Assembly.CreateOutputAssembly(Context);
}
}
}
/// <summary>
/// Bind comments to this symbol.
/// Comments are only bound to source declarations.
/// </summary>
protected void BindComments()
{
if (!symbol.IsImplicitlyDeclared && IsSourceDeclaration && symbol.FromSource())
Context.BindComments(this, FullLocation);
}
public BlockSyntax Block
{
get
{
return symbol.
DeclaringSyntaxReferences.
Select(r => r.GetSyntax()).
SelectMany(n => n.ChildNodes()).
OfType<BlockSyntax>().
FirstOrDefault();
}
}
public ExpressionSyntax ExpressionBody
{
get
{
return symbol.
DeclaringSyntaxReferences.
SelectMany(r => r.GetSyntax().ChildNodes()).
OfType<ArrowExpressionClauseSyntax>().
Select(arrow => arrow.Expression).
FirstOrDefault();
}
}
public virtual bool IsSourceDeclaration => symbol.IsSourceDeclaration();
public override bool NeedsPopulation => Context.Defines(symbol);
public Extraction.Entities.Location Location => Context.Create(ReportingLocation);
}
}