mirror of
https://github.com/github/codeql.git
synced 2025-12-17 09:13:20 +01:00
128 lines
3.5 KiB
C#
128 lines
3.5 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
|
|
namespace Semmle.Extraction.CIL
|
|
{
|
|
/// <summary>
|
|
/// Something that is extracted from an entity.
|
|
/// </summary>
|
|
///
|
|
/// <remarks>
|
|
/// The extraction algorithm proceeds as follows:
|
|
/// - Construct entity
|
|
/// - Call Extract()
|
|
/// - ILabelledEntity check if already extracted
|
|
/// - Enumerate Contents to produce more extraction products
|
|
/// - Extract these until there is nothing left to extract
|
|
/// </remarks>
|
|
public interface IExtractionProduct
|
|
{
|
|
/// <summary>
|
|
/// Perform further extraction/population of this item as necessary.
|
|
/// </summary>
|
|
///
|
|
/// <param name="cx">The extraction context.</param>
|
|
void Extract(Context cx);
|
|
}
|
|
|
|
/// <summary>
|
|
/// An entity which has been extracted.
|
|
/// </summary>
|
|
public interface IExtractedEntity : IEntity, IExtractionProduct
|
|
{
|
|
/// <summary>
|
|
/// The contents of the entity.
|
|
/// </summary>
|
|
IEnumerable<IExtractionProduct> Contents { get; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// An entity that has contents to extract. There is no need to populate
|
|
/// a key as it's done in the contructor.
|
|
/// </summary>
|
|
public abstract class UnlabelledEntity : IExtractedEntity
|
|
{
|
|
public abstract IEnumerable<IExtractionProduct> Contents { get; }
|
|
public Label Label { get; set; }
|
|
|
|
public Microsoft.CodeAnalysis.Location ReportingLocation => throw new NotImplementedException();
|
|
|
|
public virtual IId Id => FreshId.Instance;
|
|
|
|
public virtual void Extract(Context cx2)
|
|
{
|
|
cx2.Extract(this);
|
|
}
|
|
|
|
public readonly Context cx;
|
|
|
|
protected UnlabelledEntity(Context cx)
|
|
{
|
|
this.cx = cx;
|
|
cx.cx.AddFreshLabel(this);
|
|
}
|
|
|
|
TrapStackBehaviour IEntity.TrapStackBehaviour => TrapStackBehaviour.NoLabel;
|
|
}
|
|
|
|
/// <summary>
|
|
/// An entity that needs to be populated during extraction.
|
|
/// This assigns a key and optionally extracts its contents.
|
|
/// </summary>
|
|
public abstract class LabelledEntity : ILabelledEntity
|
|
{
|
|
public abstract IEnumerable<IExtractionProduct> Contents { get; }
|
|
public Label Label { get; set; }
|
|
public Microsoft.CodeAnalysis.Location ReportingLocation => throw new NotImplementedException();
|
|
|
|
public Id ShortId { get; set; }
|
|
public abstract Id IdSuffix { get; }
|
|
public IId Id => ShortId + IdSuffix;
|
|
|
|
public void Extract(Context cx2)
|
|
{
|
|
cx2.Populate(this);
|
|
}
|
|
|
|
public readonly Context cx;
|
|
|
|
protected LabelledEntity(Context cx)
|
|
{
|
|
this.cx = cx;
|
|
}
|
|
|
|
public override string ToString() => Id.ToString();
|
|
|
|
TrapStackBehaviour IEntity.TrapStackBehaviour => TrapStackBehaviour.NoLabel;
|
|
}
|
|
|
|
/// <summary>
|
|
/// An entity with a defined ID.
|
|
/// </summary>
|
|
public interface ILabelledEntity : IExtractedEntity
|
|
{
|
|
Id ShortId { get; set; }
|
|
Id IdSuffix { get; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// A tuple that is an extraction product.
|
|
/// </summary>
|
|
class Tuple : IExtractionProduct
|
|
{
|
|
readonly Extraction.Tuple tuple;
|
|
|
|
public Tuple(string name, params object[] args)
|
|
{
|
|
tuple = new Extraction.Tuple(name, args);
|
|
}
|
|
|
|
public void Extract(Context cx)
|
|
{
|
|
cx.cx.Emit(tuple);
|
|
}
|
|
|
|
public override string ToString() => tuple.ToString();
|
|
}
|
|
}
|