feat: Grammar Parser (#3)
Reviewed-on: https://git.bupt-hpc.cn/jackfiled/CanonSharp/pulls/3 Co-authored-by: jackfiled <xcrenchangjun@outlook.com> Co-committed-by: jackfiled <xcrenchangjun@outlook.com>
This commit is contained in:
@@ -1,49 +0,0 @@
|
||||
namespace CanonSharp.Combinator.Abstractions;
|
||||
|
||||
/// <summary>
|
||||
/// 失败解析结果基类
|
||||
/// </summary>
|
||||
/// <typeparam name="TToken">输入流类型</typeparam>
|
||||
/// <typeparam name="T">解析结果类型</typeparam>
|
||||
public abstract class FailedResult<TToken, T> : ParseResult<TToken, T>
|
||||
{
|
||||
public override T Value => throw Exception;
|
||||
|
||||
/// <summary>
|
||||
/// 当前读取到的状态
|
||||
/// </summary>
|
||||
public abstract IReadState<TToken> State { get; }
|
||||
|
||||
/// <summary>
|
||||
/// 解析失败的消息
|
||||
/// </summary>
|
||||
public abstract string Message { get; }
|
||||
|
||||
/// <summary>
|
||||
/// 解析失败的异常
|
||||
/// </summary>
|
||||
public virtual ParseException Exception => new(ToString());
|
||||
|
||||
/// <summary>
|
||||
/// 转换该失败结果的类型
|
||||
/// </summary>
|
||||
/// <typeparam name="TNext">转换之后的结果类型</typeparam>
|
||||
/// <returns>转换之后的失败解析类型</returns>
|
||||
public abstract FailedResult<TToken, TNext> Convert<TNext>();
|
||||
|
||||
internal override ParseResult<TToken, TResult> Next<TNext, TResult>(Func<T, Parser<TToken, TNext>> nextParser,
|
||||
Func<ParseResult<TToken, TNext>, ParseResult<TToken, TResult>> continuation)
|
||||
=> continuation(Convert<TNext>());
|
||||
|
||||
public override ParseResult<TToken, TResult> Map<TResult>(Func<T, TResult> map)
|
||||
=> Convert<TResult>();
|
||||
|
||||
public override TResult CaseOf<TResult>(Func<SuccessfulResult<TToken, T>, TResult> successfulHandler,
|
||||
Func<FailedResult<TToken, T>, TResult> failedHandler)
|
||||
=> failedHandler(this);
|
||||
|
||||
public override string ToString()
|
||||
{
|
||||
return $"Parse Failed: {Message}.";
|
||||
}
|
||||
}
|
45
CanonSharp.Combinator/Abstractions/IFailedResult.cs
Normal file
45
CanonSharp.Combinator/Abstractions/IFailedResult.cs
Normal file
@@ -0,0 +1,45 @@
|
||||
namespace CanonSharp.Combinator.Abstractions;
|
||||
|
||||
/// <summary>
|
||||
/// 失败解析结果基类
|
||||
/// </summary>
|
||||
/// <typeparam name="TToken">输入流类型</typeparam>
|
||||
/// <typeparam name="T">解析结果类型</typeparam>
|
||||
public interface IFailedResult<TToken, out T> : IParseResult<TToken, T>
|
||||
{
|
||||
T IParseResult<TToken, T>.Value => throw Exception;
|
||||
|
||||
/// <summary>
|
||||
/// 当前读取到的状态
|
||||
/// </summary>
|
||||
public IReadState<TToken> State { get; }
|
||||
|
||||
/// <summary>
|
||||
/// 解析失败的消息
|
||||
/// </summary>
|
||||
public string Message { get; }
|
||||
|
||||
/// <summary>
|
||||
/// 解析失败的异常
|
||||
/// </summary>
|
||||
public ParseException Exception => new(Message);
|
||||
|
||||
/// <summary>
|
||||
/// 转换该失败结果的类型
|
||||
/// </summary>
|
||||
/// <typeparam name="TNext">转换之后的结果类型</typeparam>
|
||||
/// <returns>转换之后的失败解析类型</returns>
|
||||
public IFailedResult<TToken, TNext> Convert<TNext>();
|
||||
|
||||
IParseResult<TToken, TResult> IParseResult<TToken, T>.Next<TNext, TResult>(
|
||||
Func<T, IParser<TToken, TNext>> nextParser,
|
||||
Func<IParseResult<TToken, TNext>, IParseResult<TToken, TResult>> continuation)
|
||||
=> continuation(Convert<TNext>());
|
||||
|
||||
IParseResult<TToken, TResult> IParseResult<TToken, T>.Map<TResult>(Func<T, TResult> map)
|
||||
=> Convert<TResult>();
|
||||
|
||||
TResult IParseResult<TToken, T>.CaseOf<TResult>(Func<ISuccessfulResult<TToken, T>, TResult> successfulHandler,
|
||||
Func<IFailedResult<TToken, T>, TResult> failedHandler)
|
||||
=> failedHandler(this);
|
||||
}
|
@@ -5,17 +5,12 @@ namespace CanonSharp.Combinator.Abstractions;
|
||||
/// </summary>
|
||||
/// <typeparam name="TToken">输入流类型</typeparam>
|
||||
/// <typeparam name="T">实际结果类型</typeparam>
|
||||
public abstract class ParseResult<TToken, T>
|
||||
public interface IParseResult<TToken, out T>
|
||||
{
|
||||
/// <summary>
|
||||
/// 实际结果对象
|
||||
/// </summary>
|
||||
public abstract T Value { get; }
|
||||
|
||||
protected ParseResult()
|
||||
{
|
||||
|
||||
}
|
||||
public T Value { get; }
|
||||
|
||||
/// <summary>
|
||||
/// 在当前结果上应用下一个解析器
|
||||
@@ -25,8 +20,8 @@ public abstract class ParseResult<TToken, T>
|
||||
/// <typeparam name="TNext">下一个解析器函数返回的解析结果类型</typeparam>
|
||||
/// <typeparam name="TResult">最终的解析结果类型</typeparam>
|
||||
/// <returns></returns>
|
||||
internal abstract ParseResult<TToken, TResult> Next<TNext, TResult>(Func<T, Parser<TToken, TNext>> nextParser,
|
||||
Func<ParseResult<TToken, TNext>, ParseResult<TToken, TResult>> continuation);
|
||||
internal IParseResult<TToken, TResult> Next<TNext, TResult>(Func<T, IParser<TToken, TNext>> nextParser,
|
||||
Func<IParseResult<TToken, TNext>, IParseResult<TToken, TResult>> continuation);
|
||||
|
||||
/// <summary>
|
||||
/// 映射结果
|
||||
@@ -34,7 +29,7 @@ public abstract class ParseResult<TToken, T>
|
||||
/// <param name="map">映射结果的函数</param>
|
||||
/// <typeparam name="TResult">映射结果函数返回解析结果的类型</typeparam>
|
||||
/// <returns>最终的解析结果</returns>
|
||||
public abstract ParseResult<TToken, TResult> Map<TResult>(Func<T, TResult> map);
|
||||
public IParseResult<TToken, TResult> Map<TResult>(Func<T, TResult> map);
|
||||
|
||||
/// <summary>
|
||||
/// 在成功或者失败解析结果上应用不同的后继函数
|
||||
@@ -43,6 +38,6 @@ public abstract class ParseResult<TToken, T>
|
||||
/// <param name="failedHandler">在失败解析结构上应用的函数</param>
|
||||
/// <typeparam name="TResult">最后返回解析结果的类型</typeparam>
|
||||
/// <returns>最后的解析结果</returns>
|
||||
public abstract TResult CaseOf<TResult>(Func<SuccessfulResult<TToken, T>, TResult> successfulHandler,
|
||||
Func<FailedResult<TToken, T>, TResult> failedHandler);
|
||||
public TResult CaseOf<TResult>(Func<ISuccessfulResult<TToken, T>, TResult> successfulHandler,
|
||||
Func<IFailedResult<TToken, T>, TResult> failedHandler);
|
||||
}
|
@@ -7,7 +7,7 @@ namespace CanonSharp.Combinator.Abstractions;
|
||||
/// </summary>
|
||||
/// <typeparam name="TToken">输入流类型</typeparam>
|
||||
/// <typeparam name="T">解析结果类型</typeparam>
|
||||
public abstract class Parser<TToken, T>
|
||||
public interface IParser<TToken, out T>
|
||||
{
|
||||
/// <summary>
|
||||
/// 解析器运行函数
|
||||
@@ -17,16 +17,16 @@ public abstract class Parser<TToken, T>
|
||||
/// <typeparam name="TState">输入流状态类型</typeparam>
|
||||
/// <typeparam name="TResult">后继函数运行之后的解析结果类型</typeparam>
|
||||
/// <returns></returns>
|
||||
internal abstract ParseResult<TToken, TResult> Run<TState, TResult>(TState state,
|
||||
Func<ParseResult<TToken, T>, ParseResult<TToken, TResult>> continuation)
|
||||
internal IParseResult<TToken, TResult> Run<TState, TResult>(TState state,
|
||||
Func<IParseResult<TToken, T>, IParseResult<TToken, TResult>> continuation)
|
||||
where TState : IReadState<TToken, TState>;
|
||||
|
||||
public ParseResult<TToken, T> Parse<TState>(TState state) where TState : IReadState<TToken, TState>
|
||||
public IParseResult<TToken, T> Parse<TState>(TState state) where TState : IReadState<TToken, TState>
|
||||
{
|
||||
return Run(state);
|
||||
}
|
||||
|
||||
private ParseResult<TToken, T> Run<TState>(TState state) where TState : IReadState<TToken, TState>
|
||||
private IParseResult<TToken, T> Run<TState>(TState state) where TState : IReadState<TToken, TState>
|
||||
{
|
||||
try
|
||||
{
|
||||
@@ -38,6 +38,6 @@ public abstract class Parser<TToken, T>
|
||||
}
|
||||
}
|
||||
|
||||
public static Parser<TToken, T> operator |(Parser<TToken, T> a, Parser<TToken, T> b)
|
||||
public static IParser<TToken, T> operator |(IParser<TToken, T> a, IParser<TToken, T> b)
|
||||
=> a.Alternative(b);
|
||||
}
|
29
CanonSharp.Combinator/Abstractions/ISuccessfulResult.cs
Normal file
29
CanonSharp.Combinator/Abstractions/ISuccessfulResult.cs
Normal file
@@ -0,0 +1,29 @@
|
||||
namespace CanonSharp.Combinator.Abstractions;
|
||||
|
||||
/// <summary>
|
||||
/// 成功解析结果基类
|
||||
/// </summary>
|
||||
/// <typeparam name="TToken">输入流类型</typeparam>
|
||||
/// <typeparam name="T">实际的解析结果类型</typeparam>
|
||||
public interface ISuccessfulResult<TToken,out T> : IParseResult<TToken, T>
|
||||
{
|
||||
/// <summary>
|
||||
/// 运行下一个解析器
|
||||
/// </summary>
|
||||
/// <param name="parser">下一个解析器</param>
|
||||
/// <param name="continuation">处理解析结果的后继函数</param>
|
||||
/// <typeparam name="TNext">下一个解析器返回的结果类型</typeparam>
|
||||
/// <typeparam name="TResult">最终的结果类型</typeparam>
|
||||
/// <returns>最终的结果</returns>
|
||||
protected IParseResult<TToken, TResult> RunNext<TNext, TResult>(IParser<TToken, TNext> parser,
|
||||
Func<IParseResult<TToken, TNext>, IParseResult<TToken, TResult>> continuation);
|
||||
|
||||
IParseResult<TToken, TResult> IParseResult<TToken, T>.Next<TNext, TResult>(
|
||||
Func<T, IParser<TToken, TNext>> nextParser,
|
||||
Func<IParseResult<TToken, TNext>, IParseResult<TToken, TResult>> continuation)
|
||||
=> RunNext(nextParser(Value), continuation);
|
||||
|
||||
TResult IParseResult<TToken, T>.CaseOf<TResult>(Func<ISuccessfulResult<TToken, T>, TResult> successfulHandler,
|
||||
Func<IFailedResult<TToken, T>, TResult> failedHandler)
|
||||
=> successfulHandler(this);
|
||||
}
|
@@ -1,33 +0,0 @@
|
||||
namespace CanonSharp.Combinator.Abstractions;
|
||||
|
||||
/// <summary>
|
||||
/// 成功解析结果基类
|
||||
/// </summary>
|
||||
/// <param name="value">实际的解析结果</param>
|
||||
/// <typeparam name="TToken">输入流类型</typeparam>
|
||||
/// <typeparam name="T">实际的解析结果类型</typeparam>
|
||||
public abstract class SuccessfulResult<TToken, T>(T value) : ParseResult<TToken, T>
|
||||
{
|
||||
public override T Value => value;
|
||||
|
||||
/// <summary>
|
||||
/// 运行下一个解析器
|
||||
/// </summary>
|
||||
/// <param name="parser">下一个解析器</param>
|
||||
/// <param name="continuation">处理解析结果的后继函数</param>
|
||||
/// <typeparam name="TNext">下一个解析器返回的结果类型</typeparam>
|
||||
/// <typeparam name="TResult">最终的结果类型</typeparam>
|
||||
/// <returns>最终的结果</returns>
|
||||
protected abstract ParseResult<TToken, TResult> RunNext<TNext, TResult>(Parser<TToken, TNext> parser,
|
||||
Func<ParseResult<TToken, TNext>, ParseResult<TToken, TResult>> continuation);
|
||||
|
||||
internal override ParseResult<TToken, TResult> Next<TNext, TResult>(Func<T, Parser<TToken, TNext>> nextParser,
|
||||
Func<ParseResult<TToken, TNext>, ParseResult<TToken, TResult>> continuation)
|
||||
=> RunNext(nextParser(Value), continuation);
|
||||
|
||||
public override TResult CaseOf<TResult>(Func<SuccessfulResult<TToken, T>, TResult> successfulHandler,
|
||||
Func<FailedResult<TToken, T>, TResult> failedHandler)
|
||||
=> successfulHandler(this);
|
||||
|
||||
public override string ToString() => Value?.ToString() ?? string.Empty;
|
||||
}
|
Reference in New Issue
Block a user