#region Copyright notice and license // Protocol Buffers - Google's data interchange format // Copyright 2008 Google Inc. All rights reserved. // // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file or at // https://developers.google.com/open-source/licenses/bsd #endregion using System; namespace Google.Protobuf { internal sealed class JsonToken : IEquatable { internal static JsonToken Null { get; } = new JsonToken(TokenType.Null); internal static JsonToken False { get; } = new JsonToken(TokenType.False); internal static JsonToken True { get; } = new JsonToken(TokenType.True); internal static JsonToken StartObject { get; } = new JsonToken(TokenType.StartObject); internal static JsonToken EndObject { get; } = new JsonToken(TokenType.EndObject); internal static JsonToken StartArray { get; } = new JsonToken(TokenType.StartArray); internal static JsonToken EndArray { get; } = new JsonToken(TokenType.EndArray); internal static JsonToken EndDocument { get; } = new JsonToken(TokenType.EndDocument); internal static JsonToken Name(string name) { return new JsonToken(TokenType.Name, stringValue: name); } internal static JsonToken Value(string value) { return new JsonToken(TokenType.StringValue, stringValue: value); } internal static JsonToken Value(double value) { return new JsonToken(TokenType.Number, numberValue: value); } internal enum TokenType { Null, False, True, StringValue, Number, Name, StartObject, EndObject, StartArray, EndArray, EndDocument } // A value is a string, number, array, object, null, true or false // Arrays and objects have start/end // A document consists of a value // Objects are name/value sequences. private readonly TokenType type; private readonly string stringValue; private readonly double numberValue; internal TokenType Type => type; internal string StringValue => stringValue; internal double NumberValue => numberValue; private JsonToken(TokenType type, string stringValue = null, double numberValue = 0) { this.type = type; this.stringValue = stringValue; this.numberValue = numberValue; } public override bool Equals(object obj) => Equals(obj as JsonToken); public override int GetHashCode() { unchecked { int hash = 17; hash = hash * 31 + (int) type; hash = hash * 31 + stringValue == null ? 0 : stringValue.GetHashCode(); hash = hash * 31 + numberValue.GetHashCode(); return hash; } } public override string ToString() { return type switch { TokenType.Null => "null", TokenType.True => "true", TokenType.False => "false", TokenType.Name => $"name ({stringValue})", TokenType.StringValue => $"value ({stringValue})", TokenType.Number => $"number ({numberValue})", TokenType.StartObject => "start-object", TokenType.EndObject => "end-object", TokenType.StartArray => "start-array", TokenType.EndArray => "end-array", TokenType.EndDocument => "end-document", _ => throw new InvalidOperationException($"Token is of unknown type {type}"), }; } public bool Equals(JsonToken other) { if (other is null) { return false; } // Note use of other.numberValue.Equals rather than ==, so that NaN compares appropriately. return other.type == type && other.stringValue == stringValue && other.numberValue.Equals(numberValue); } } }