forked from mirrors/NBTExplorer
386 lines
13 KiB
C#
386 lines
13 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using Substrate.Core;
|
|
|
|
namespace Substrate.NBT
|
|
{
|
|
/// <summary>
|
|
/// Indicates how an <see cref="NBTVerifier"/> event processor should respond to returning event handler.
|
|
/// </summary>
|
|
public enum TagEventCode
|
|
{
|
|
/// <summary>
|
|
/// The event processor should process the next event in the chian.
|
|
/// </summary>
|
|
NEXT,
|
|
|
|
/// <summary>
|
|
/// The event processor should ignore the verification failure and stop processing any remaining events.
|
|
/// </summary>
|
|
PASS,
|
|
|
|
/// <summary>
|
|
/// The event processor should fail and stop processing any remaining events.
|
|
/// </summary>
|
|
FAIL,
|
|
}
|
|
|
|
/// <summary>
|
|
/// Event arguments for <see cref="NBTVerifier"/> failure events.
|
|
/// </summary>
|
|
public class TagEventArgs : EventArgs
|
|
{
|
|
private string _tagName;
|
|
private TagNode _parent;
|
|
private TagNode _tag;
|
|
private SchemaNode _schema;
|
|
|
|
/// <summary>
|
|
/// Gets the expected name of the <see cref="TagNode"/> referenced by this event.
|
|
/// </summary>
|
|
public string TagName
|
|
{
|
|
get { return _tagName; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the parent <see cref="TagNode"/> of the <see cref="TagNode"/> referenced by this event, if it exists.
|
|
/// </summary>
|
|
public TagNode Parent
|
|
{
|
|
get { return _parent; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the <see cref="TagNode"/> referenced by this event.
|
|
/// </summary>
|
|
public TagNode Tag
|
|
{
|
|
get { return _tag; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the <see cref="SchemaNode"/> corresponding to the <see cref="TagNode"/> referenced by this event.
|
|
/// </summary>
|
|
public SchemaNode Schema
|
|
{
|
|
get { return _schema; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructs a new event argument set.
|
|
/// </summary>
|
|
/// <param name="tagName">The expected name of a <see cref="TagNode"/>.</param>
|
|
public TagEventArgs (string tagName)
|
|
: base()
|
|
{
|
|
_tagName = tagName;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructs a new event argument set.
|
|
/// </summary>
|
|
/// <param name="tagName">The expected name of a <see cref="TagNode"/>.</param>
|
|
/// <param name="tag">The <see cref="TagNode"/> involved in this event.</param>
|
|
public TagEventArgs (string tagName, TagNode tag)
|
|
: base()
|
|
{
|
|
_tag = tag;
|
|
_tagName = tagName;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructs a new event argument set.
|
|
/// </summary>
|
|
/// <param name="schema">The <see cref="SchemaNode"/> corresponding to the <see cref="TagNode"/> involved in this event.</param>
|
|
/// <param name="tag">The <see cref="TagNode"/> involved in this event.</param>
|
|
public TagEventArgs (SchemaNode schema, TagNode tag)
|
|
: base()
|
|
{
|
|
_tag = tag;
|
|
_schema = schema;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// An event handler for intercepting and responding to verification failures of NBT trees.
|
|
/// </summary>
|
|
/// <param name="eventArgs">Information relating to a verification event.</param>
|
|
/// <returns>A <see cref="TagEventCode"/> determining how the event processor should respond.</returns>
|
|
public delegate TagEventCode VerifierEventHandler (TagEventArgs eventArgs);
|
|
|
|
/// <summary>
|
|
/// Verifies the integrity of an NBT tree against a schema definition.
|
|
/// </summary>
|
|
public class NBTVerifier
|
|
{
|
|
private TagNode _root;
|
|
private SchemaNode _schema;
|
|
|
|
/// <summary>
|
|
/// An event that gets fired whenever an expected <see cref="TagNode"/> is not found.
|
|
/// </summary>
|
|
public static event VerifierEventHandler MissingTag;
|
|
|
|
/// <summary>
|
|
/// An event that gets fired whenever an expected <see cref="TagNode"/> is of the wrong type and cannot be cast.
|
|
/// </summary>
|
|
public static event VerifierEventHandler InvalidTagType;
|
|
|
|
/// <summary>
|
|
/// An event that gets fired whenever an expected <see cref="TagNode"/> has a value that violates the schema.
|
|
/// </summary>
|
|
public static event VerifierEventHandler InvalidTagValue;
|
|
|
|
private Dictionary<string, TagNode> _scratch = new Dictionary<string,TagNode>();
|
|
|
|
/// <summary>
|
|
/// Constructs a new <see cref="NBTVerifier"/> object for a given NBT tree and schema.
|
|
/// </summary>
|
|
/// <param name="root">A <see cref="TagNode"/> representing the root of an NBT tree.</param>
|
|
/// <param name="schema">A <see cref="SchemaNode"/> representing the root of a schema definition for the NBT tree.</param>
|
|
public NBTVerifier (TagNode root, SchemaNode schema)
|
|
{
|
|
_root = root;
|
|
_schema = schema;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Invokes the verifier.
|
|
/// </summary>
|
|
/// <returns>Status indicating whether the NBT tree is valid for the given schema.</returns>
|
|
public virtual bool Verify ()
|
|
{
|
|
return Verify(null, _root, _schema);
|
|
}
|
|
|
|
private bool Verify (TagNode parent, TagNode tag, SchemaNode schema)
|
|
{
|
|
if (tag == null) {
|
|
return OnMissingTag(new TagEventArgs(schema.Name));
|
|
}
|
|
|
|
SchemaNodeScaler scaler = schema as SchemaNodeScaler;
|
|
if (scaler != null) {
|
|
return VerifyScaler(tag, scaler);
|
|
}
|
|
|
|
SchemaNodeString str = schema as SchemaNodeString;
|
|
if (str != null) {
|
|
return VerifyString(tag, str);
|
|
}
|
|
|
|
SchemaNodeArray array = schema as SchemaNodeArray;
|
|
if (array != null) {
|
|
return VerifyArray(tag, array);
|
|
}
|
|
|
|
SchemaNodeList list = schema as SchemaNodeList;
|
|
if (list != null) {
|
|
return VerifyList(tag, list);
|
|
}
|
|
|
|
SchemaNodeCompound compound = schema as SchemaNodeCompound;
|
|
if (compound != null) {
|
|
return VerifyCompound(tag, compound);
|
|
}
|
|
|
|
return OnInvalidTagType(new TagEventArgs(schema.Name, tag));
|
|
}
|
|
|
|
private bool VerifyScaler (TagNode tag, SchemaNodeScaler schema)
|
|
{
|
|
if (!tag.IsCastableTo(schema.Type)) {
|
|
if (!OnInvalidTagType(new TagEventArgs(schema.Name, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private bool VerifyString (TagNode tag, SchemaNodeString schema)
|
|
{
|
|
TagNodeString stag = tag as TagNodeString;
|
|
if (stag == null) {
|
|
if (!OnInvalidTagType(new TagEventArgs(schema, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
if (schema.Length > 0 && stag.Length > schema.Length) {
|
|
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
if (schema.Value != null && stag.Data != schema.Value) {
|
|
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
private bool VerifyArray (TagNode tag, SchemaNodeArray schema)
|
|
{
|
|
TagNodeByteArray atag = tag as TagNodeByteArray;
|
|
if (atag == null) {
|
|
if (!OnInvalidTagType(new TagEventArgs(schema, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
if (schema.Length > 0 && atag.Length != schema.Length) {
|
|
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private bool VerifyList (TagNode tag, SchemaNodeList schema)
|
|
{
|
|
TagNodeList ltag = tag as TagNodeList;
|
|
if (ltag == null) {
|
|
if (!OnInvalidTagType(new TagEventArgs(schema, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
if (ltag.Count > 0 && ltag.ValueType != schema.Type) {
|
|
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
if (schema.Length > 0 && ltag.Count != schema.Length) {
|
|
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Patch up empty lists
|
|
//if (schema.Length == 0) {
|
|
// tag = new NBT_List(schema.Type);
|
|
//}
|
|
|
|
bool pass = true;
|
|
|
|
// If a subschema is set, test all items in list against it
|
|
|
|
if (schema.SubSchema != null) {
|
|
foreach (TagNode v in ltag) {
|
|
pass = Verify(tag, v, schema.SubSchema) && pass;
|
|
}
|
|
}
|
|
|
|
return pass;
|
|
}
|
|
|
|
private bool VerifyCompound (TagNode tag, SchemaNodeCompound schema)
|
|
{
|
|
TagNodeCompound ctag = tag as TagNodeCompound;
|
|
if (ctag == null) {
|
|
if (!OnInvalidTagType(new TagEventArgs(schema, tag))) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool pass = true;
|
|
|
|
foreach (SchemaNode node in schema) {
|
|
TagNode value;
|
|
ctag.TryGetValue(node.Name, out value);
|
|
|
|
if (value == null) {
|
|
if ((node.Options & SchemaOptions.CREATE_ON_MISSING) == SchemaOptions.CREATE_ON_MISSING) {
|
|
_scratch[node.Name] = node.BuildDefaultTree();
|
|
continue;
|
|
}
|
|
else if ((node.Options & SchemaOptions.OPTIONAL) == SchemaOptions.OPTIONAL) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
pass = Verify(tag, value, node) && pass;
|
|
}
|
|
|
|
foreach (KeyValuePair<string, TagNode> item in _scratch) {
|
|
ctag[item.Key] = item.Value;
|
|
}
|
|
|
|
_scratch.Clear();
|
|
|
|
return pass;
|
|
}
|
|
|
|
#region Event Handlers
|
|
|
|
/// <summary>
|
|
/// Processes registered events for <see cref="MissingTag"/> whenever an expected <see cref="TagNode"/> is not found.
|
|
/// </summary>
|
|
/// <param name="e">Arguments for this event.</param>
|
|
/// <returns>Status indicating whether this event can be ignored.</returns>
|
|
protected virtual bool OnMissingTag (TagEventArgs e)
|
|
{
|
|
if (MissingTag != null) {
|
|
foreach (VerifierEventHandler func in MissingTag.GetInvocationList()) {
|
|
TagEventCode code = func(e);
|
|
switch (code) {
|
|
case TagEventCode.FAIL:
|
|
return false;
|
|
case TagEventCode.PASS:
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Processes registered events for <see cref="InvalidTagType"/> whenever an expected <see cref="TagNode"/> is of the wrong type and cannot be cast.
|
|
/// </summary>
|
|
/// <param name="e">Arguments for this event.</param>
|
|
/// <returns>Status indicating whether this event can be ignored.</returns>
|
|
protected virtual bool OnInvalidTagType (TagEventArgs e)
|
|
{
|
|
if (InvalidTagType != null) {
|
|
foreach (VerifierEventHandler func in InvalidTagType.GetInvocationList()) {
|
|
TagEventCode code = func(e);
|
|
switch (code) {
|
|
case TagEventCode.FAIL:
|
|
return false;
|
|
case TagEventCode.PASS:
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Processes registered events for <see cref="InvalidTagValue"/> whenever an expected <see cref="TagNode"/> has a value that violates the schema.
|
|
/// </summary>
|
|
/// <param name="e">Arguments for this event.</param>
|
|
/// <returns>Status indicating whether this event can be ignored.</returns>
|
|
protected virtual bool OnInvalidTagValue (TagEventArgs e)
|
|
{
|
|
if (InvalidTagValue != null) {
|
|
foreach (VerifierEventHandler func in InvalidTagValue.GetInvocationList()) {
|
|
TagEventCode code = func(e);
|
|
switch (code) {
|
|
case TagEventCode.FAIL:
|
|
return false;
|
|
case TagEventCode.PASS:
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|