Files
crtr/Assets/YamlDotNet/Core/ParserExtensions.cs
2022-12-12 22:23:49 +08:00

152 lines
5.9 KiB
C#

// This file is part of YamlDotNet - A .NET library for YAML.
// Copyright (c) Antoine Aubry and contributors
//
// Permission is hereby granted, free of charge, to any person obtaining a copy of
// this software and associated documentation files (the "Software"), to deal in
// the Software without restriction, including without limitation the rights to
// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
// of the Software, and to permit persons to whom the Software is furnished to do
// so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
using System;
using System.Diagnostics.CodeAnalysis;
using System.IO;
using YamlDotNet.Core.Events;
namespace YamlDotNet.Core
{
/// <summary>
/// Extension methods that provide useful abstractions over <see cref="IParser"/>.
/// </summary>
public static class ParserExtensions
{
/// <summary>
/// Ensures that the current event is of the specified type, returns it and moves to the next event.
/// </summary>
/// <typeparam name="T">Type of the <see cref="ParsingEvent"/>.</typeparam>
/// <returns>Returns the current event.</returns>
/// <exception cref="YamlException">If the current event is not of the specified type.</exception>
public static T Consume<T>(this IParser parser) where T : ParsingEvent
{
var required = parser.Require<T>();
parser.MoveNext();
return required;
}
/// <summary>
/// Checks whether the current event is of the specified type.
/// If the event is of the specified type, returns it and moves to the next event.
/// Otherwise returns null.
/// </summary>
/// <typeparam name="T">Type of the <see cref="ParsingEvent"/>.</typeparam>
/// <returns>Returns true if the current event is of type T; otherwise returns null.</returns>
public static bool TryConsume<T>(this IParser parser, [MaybeNullWhen(false)] out T @event) where T : ParsingEvent
{
if (parser.Accept(out @event!))
{
parser.MoveNext();
return true;
}
return false;
}
/// <summary>
/// Enforces that the current event is of the specified type.
/// </summary>
/// <typeparam name="T">Type of the <see cref="ParsingEvent"/>.</typeparam>
/// <returns>Returns the current event.</returns>
/// <exception cref="YamlException">If the current event is not of the specified type.</exception>
public static T Require<T>(this IParser parser) where T : ParsingEvent
{
if (!parser.Accept<T>(out var required))
{
var @event = parser.Current;
if (@event == null)
{
throw new YamlException($"Expected '{typeof(T).Name}', got nothing.");
}
throw new YamlException(@event.Start, @event.End, $"Expected '{typeof(T).Name}', got '{@event.GetType().Name}' (at {@event.Start}).");
}
return required;
}
/// <summary>
/// Checks whether the current event is of the specified type.
/// </summary>
/// <typeparam name="T">Type of the event.</typeparam>
/// <returns>Returns true if the current event is of type <typeparamref name="T"/>. Otherwise returns false.</returns>
public static bool Accept<T>(this IParser parser, [MaybeNullWhen(false)] out T @event) where T : ParsingEvent
{
if (parser.Current == null)
{
if (!parser.MoveNext())
{
throw new EndOfStreamException();
}
}
if (parser.Current is T evt)
{
@event = evt;
return true;
}
else
{
@event = default!;
return false;
}
}
/// <summary>
/// Skips the current event and any nested event.
/// </summary>
public static void SkipThisAndNestedEvents(this IParser parser)
{
var depth = 0;
do
{
var next = parser.Consume<ParsingEvent>();
depth += next.NestingIncrease;
}
while (depth > 0);
}
[Obsolete("Please use Consume<T>() instead")]
public static T Expect<T>(this IParser parser) where T : ParsingEvent
{
return parser.Consume<T>();
}
[Obsolete("Please use TryConsume<T>(out var evt) instead")]
[return: MaybeNull]
public static T Allow<T>(this IParser parser) where T : ParsingEvent
{
return parser.TryConsume<T>(out var @event) ? @event : default;
}
[Obsolete("Please use Accept<T>(out var evt) instead")]
[return: MaybeNull]
public static T Peek<T>(this IParser parser) where T : ParsingEvent
{
return parser.Accept<T>(out var @event) ? @event : default;
}
[Obsolete("Please use TryConsume<T>(out var evt) or Accept<T>(out var evt) instead")]
public static bool Accept<T>(this IParser parser) where T : ParsingEvent
{
return Accept<T>(parser, out var _);
}
}
}