1012 lines
36 KiB
C#
Raw Normal View History

//------------------------------------------------------------------------------
// <copyright file="LOSFormatter.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
//------------------------------------------------------------------------------
#if OBJECTSTATEFORMATTER
namespace System.Web.UI {
using System;
using System.IO;
using System.Text;
/// <devdoc>
/// Serializes Web Froms view state. The limited object serialization (LOS)
/// formatter is designed for ASCII format serialization. This class
/// supports serializing any object graph, but is optimized for those containing
/// strings, arrays, and hashtables. It offers second order optimization for many of
/// the .NET primitive types.
/// This class has been replaced with a more optimal serialization mechanism implemented
/// in LosSerializer. LosFormatter itself uses LosSerialization as part of its
/// implementation to benefit from the highly compact serialization when possible.
/// </devdoc>
public sealed class LosFormatter {
private const int InitialBufferSize = 24;
private ObjectStateFormatter _formatter;
private bool _enableMac;
/// <devdoc>
/// <para>Creates a LosFormatter object.</para>
/// </devdoc>
public LosFormatter() : this(false, (byte[])null) {
}
/// <devdoc>
/// <para>Creates a LosFormatter object, specifying whether view state mac should be
/// enabled. If it is, use macKeyModifier to modify the mac key.</para>
/// </devdoc>
public LosFormatter(bool enableMac, string macKeyModifier): this (enableMac, GetBytes(macKeyModifier)) {
}
public LosFormatter(bool enableMac, byte[] macKeyModifier) {
_enableMac = enableMac;
if (enableMac) {
_formatter = new ObjectStateFormatter(macKeyModifier);
}
else {
_formatter = new ObjectStateFormatter();
}
}
private static byte[] GetBytes(string s) {
if (s != null && s.Length != 0)
return Encoding.Unicode.GetBytes(s);
else
return null;
}
/// <devdoc>
/// <para> Deserializes a LOS-formatted object from a <see cref='System.IO.Stream'/> object.</para>
/// </devdoc>
public object Deserialize(Stream stream) {
TextReader input = null;
input = new StreamReader(stream);
return Deserialize(input);
}
/// <devdoc>
/// <para>Deserializes a LOS-formatted object from a <see cref='System.IO.TextReader'/> object.</para>
/// </devdoc>
public object Deserialize(TextReader input) {
char[] data = new char[128];
int read = 0;
int current = 0;
int blockSize = InitialBufferSize;
do {
read = input.Read(data, current, blockSize);
current += read;
if (current > data.Length - blockSize) {
char[] bigger = new char[data.Length * 2];
Array.Copy(data, bigger, data.Length);
data = bigger;
}
} while (read == blockSize);
return Deserialize(new String(data, 0, current));
}
/// <devdoc>
/// <para>Deserializes a LOS formatted object from a string.</para>
/// </devdoc>
public object Deserialize(string input) {
return _formatter.Deserialize(input);
}
/// <devdoc>
/// <para>Serializes the Web Forms view state value into
/// a <see cref='System.IO.Stream'/> object.</para>
/// </devdoc>
public void Serialize(Stream stream, object value) {
TextWriter output = new StreamWriter(stream);
SerializeInternal(output, value);
output.Flush();
}
/// <devdoc>
/// <para>Serializes the view state value into a <see cref='System.IO.TextWriter'/> object.</para>
/// </devdoc>
public void Serialize(TextWriter output, object value) {
SerializeInternal(output, value);
}
/// <devdoc>
/// Serialized value into the writer.
/// </devdoc>
private void SerializeInternal(TextWriter output, object value) {
string data = _formatter.Serialize(value);
output.Write(data);
}
}
}
#else // !OBJECTSTATEFORMATTER
// uncomment for "human readable" debugging output - no base64 encoding.
//#define NO_BASE64
namespace System.Web.UI {
using System.Runtime.Serialization.Formatters.Binary;
using System.Runtime.Serialization;
using System;
using System.IO;
using System.Security.Principal;
using System.Collections;
using System.Collections.Specialized;
using System.Diagnostics;
using System.ComponentModel;
using System.Globalization;
using System.Threading;
using System.Text;
using System.Web.Configuration;
using System.Security.Permissions;
/// <devdoc>
/// <para>Serializes Web Froms view state. The limited object serialization (LOS)
/// formatter is designed for highly compact ASCII format serialization. This class
/// supports serializing any object graph, but is optimized for those containing
/// strings, arrays, and hashtables. It offers second order optimization for many of
/// the .NET primitive types.</para>
/// </devdoc>
public sealed class LosFormatter : IStateFormatter {
// NOTE : This formatter is not very fault tolerant, by design. I want
// : to avoid a bunch of reduntant checking... Since the format
// : is short lived we shouldn't have to worry about it.
//
// NOTE : Although hex encoding of numbers would be more effecient, it
// : would make it much harder to determine what are numbers vs.
// : names & types. Unless this becomes a problem, I suggest we
// : keep encoding in decimal.
//
// Known Types. There can only be 50 of these... The order shouldn't
// matter, we store and index into this array... although there is a
// slight perf advantage being at the top of the list...
//
private static readonly Type[] knownTypes = new Type[]
{
typeof(object),
typeof(System.Web.UI.WebControls.Unit),
typeof(System.Drawing.Color),
typeof(System.Int16),
typeof(System.Int64),
};
static readonly Encoding EncodingInstance = new UTF8Encoding(false);
static readonly NumberFormatInfo NumberFormat = NumberFormatInfo.InvariantInfo;
private const int UntypedTypeId = -1;
private const int NoTypeId = -2;
private const int InitialBufferSize = 24;
private const int BufferGrowth = 48;
// Constant chars and strings... you can change these and all references to the
// begin, end, and delimiter chars are fixed up
private const char leftAngleBracketChar = '<';
private const char rightAngleBracketChar = '>';
private const char valueDelimiterChar = ';';
private static readonly char[] escapedCharacters = { leftAngleBracketChar, rightAngleBracketChar, valueDelimiterChar, '\\' };
private static CharBufferAllocator _charBufferAllocator = new CharBufferAllocator(256, 16);
// reusable Temp buffer used for constructing strings from char arrays... this is
// more performant than using a StringBuilder.
//
private char[] _builder;
private bool _recyclable;
// Tables used to build up the type and name tables during
// serialization. Not used during deserilization.
private IDictionary _typeTable;
// Deserialization variables. Not used during serialization.
private ArrayList _deserializedTypeTable;
private ListDictionary _deserializedConverterTable;
private char[] _deserializationData;
private int _current;
// MAC authentication
private bool _enableViewStateMac;
private bool EnableViewStateMac {
get { return _enableViewStateMac; }
}
private byte [] _macKey = null;
/// <devdoc>
/// <para>Creates a LosFormatter object.</para>
/// </devdoc>
public LosFormatter() {}
/// <devdoc>
/// <para>Creates a LosFormatter object, specifying whether view state mac should be
/// enabled. If it is, use macKeyModifier to modify the mac key.</para>
/// </devdoc>
public LosFormatter(bool enableMac, string macKeyModifier) {
_enableViewStateMac = enableMac;
if (macKeyModifier != null)
_macKey = Encoding.Unicode.GetBytes(macKeyModifier);
}
/// <devdoc>
/// <para> Deserializes a LOS-formatted object from a <see cref='System.IO.Stream'/> object.</para>
/// </devdoc>
public object Deserialize(Stream stream) {
TextReader input = null;
input = new StreamReader(stream);
return Deserialize(input);
}
/// <devdoc>
/// <para>Deserializes a LOS-formatted object from a <see cref='System.IO.TextReader'/> object.</para>
/// </devdoc>
public object Deserialize(TextReader input) {
char[] data = new char[128];
int read = 0;
int current = 0;
int blockSize = InitialBufferSize;
do {
read = input.Read(data, current, blockSize);
current += read;
if (current > data.Length - blockSize) {
char[] bigger = new char[data.Length * 2];
Array.Copy(data, bigger, data.Length);
data = bigger;
}
} while (read == blockSize);
return Deserialize(new String(data, 0, current));
}
/// <devdoc>
/// <para>Deserializes a LOS formatted object from a string.</para>
/// </devdoc>
public object Deserialize(string input) {
#if NO_BASE64
char[] data = input.ToCharArray();
#else
byte[] dataBytes = Convert.FromBase64String(input);
int dataLength = -1;
if (EnableViewStateMac) {
try {
dataBytes = MachineKeySection.GetDecodedData(dataBytes, _macKey, 0, dataBytes.Length, ref dataLength);
}
catch (Exception e) {
PerfCounters.IncrementCounter(AppPerfCounter.VIEWSTATE_MAC_FAIL);
ViewStateException.ThrowMacValidationError(e, input);
}
}
if (dataLength == -1) {
dataLength = dataBytes.Length;
}
char[] data = EncodingInstance.GetChars(dataBytes, 0, dataLength);
#endif
// clear or allocate name and type tables.
//
if (_deserializedTypeTable == null) {
_deserializedTypeTable = new ArrayList();
_deserializedConverterTable = new ListDictionary();
}
else {
_deserializedTypeTable.Clear();
_deserializedConverterTable.Clear();
}
_builder = (char[]) _charBufferAllocator.GetBuffer();
_recyclable = true;
// DeserializeValueInternal is recursive, so we just kick this off
// starting at 0
_current = 0;
_deserializationData = data;
object ret = DeserializeValueInternal();
if (_recyclable)
_charBufferAllocator.ReuseBuffer(_builder);
return ret;
}
/// <devdoc>
/// Deserializes a value from tokens, starting at current. When this
/// function returns, current will be left at the next token.
///
/// This function is recursive.
/// </devdoc>
private object DeserializeValueInternal() {
// Determine the data type... possible combinations are:
//
// @<...> == array of strings
// @T<...> == array of (typeref T)
// b<...> == base64 encoded value
// h<...> == hashtable
// l<...> == arraylist
// p<...> == pair
// t<...> == triplet
// i<...> == int
// o<t/f> == boolean true/false
// T<...> == (typeref T)
// ... == string
//
object value = null;
string token = ConsumeOneToken();
if (_current >= _deserializationData.Length || _deserializationData[_current] != leftAngleBracketChar) {
// just a string - next token is not a left angle bracket
// we can shortcut here and just return the string
//_current++; //consume right angle bracket or delimiter
return token;
}
_current++; // consume left angle bracket
// otherwise, we have typeref followed by value
if (token.Length == 1) {
// simple type we recognize
char ch = token[0];
if (ch == 'p') {
Pair p = new Pair();
if (_deserializationData[_current] != valueDelimiterChar) {
p.First = DeserializeValueInternal();
}
_current++; // consume delimeter
if (_deserializationData[_current] != rightAngleBracketChar) {
p.Second = DeserializeValueInternal();
}
value = p;
}
else if (ch == 't') {
Triplet t = new Triplet();
if (_deserializationData[_current] != valueDelimiterChar) {
t.First = DeserializeValueInternal();
}
_current++; // consume delimeter
if (_deserializationData[_current] != valueDelimiterChar) {
t.Second = DeserializeValueInternal();
}
_current++; // consume delimeter
if (_deserializationData[_current] != rightAngleBracketChar) {
t.Third = DeserializeValueInternal();
}
value = t;
}
// Parse int32...
else if (ch == 'i') {
value = Int32.Parse(ConsumeOneToken(), NumberFormat);
}
else if (ch == 'o') {
value = _deserializationData[_current] == 't';
_current++; // consume t or f
}
// Parse arrayList...
//
else if (ch == 'l') {
ArrayList data = new ArrayList();
while (_deserializationData[_current] != rightAngleBracketChar) {
object itemValue = null;
if (_deserializationData[_current] != valueDelimiterChar) {
itemValue = DeserializeValueInternal();
}
data.Add(itemValue);
_current++; //consume the delimiter
}
value = data;
}
else if (ch == '@') {
// if we're here, length == 1 so this is a string array
value = ConsumeStringArray();
}
// Parse hashtable...
//
else if (ch == 'h') {
Hashtable data = new Hashtable();
while (_deserializationData[_current] != rightAngleBracketChar) {
object key;
key = DeserializeValueInternal(); // hashtable key cannot be null
_current++; // consume delimiter
if (_deserializationData[_current] != valueDelimiterChar) {
data[key] = DeserializeValueInternal();
}
else {
data[key] = null;
}
_current++; // consume delimiter
}
value = data;
}
// base64 encoded...
//
else if (ch == 'b') {
string text = ConsumeOneToken();
byte[] serializedData;
serializedData = Convert.FromBase64String(text);
if (!String.IsNullOrEmpty(serializedData)) {
System.Runtime.Serialization.IFormatter formatter = new System.Runtime.Serialization.Formatters.Binary.BinaryFormatter();
value = formatter.Deserialize(new MemoryStream(serializedData));
}
}
// Parse typeconverter value ...
//
else {
// we have a typeref which is only one character long
value = ConsumeTypeConverterValue(token);
}
}
else {
// length > 1
// Parse array...
//
if (token[0] == '@') {
// if we're here, length > 1 so we must have a type ref after the @
Type creatableType = TypeFromTypeRef(token.Substring(1));
value = ConsumeArray(creatableType);
}
// Parse typeconverter value ...
//
else {
// we have a typeref which is more than one character long
value = ConsumeTypeConverterValue(token);
}
}
_current++; //consume right angle bracket
return value;
}
private string ConsumeOneToken() {
int locInBuilder = 0;
while (_current < _deserializationData.Length)
{
switch (_deserializationData[_current]) {
case '\\':
_current++; // skip slash
if (_deserializationData[_current] == 'e') {
_current++;
return String.Empty;
}
_builder[locInBuilder] = _deserializationData[_current];
locInBuilder++;
break;
case valueDelimiterChar:
case leftAngleBracketChar:
case rightAngleBracketChar:
return new string(_builder, 0, locInBuilder);
default:
_builder[locInBuilder] = _deserializationData[_current];
locInBuilder++;
break;
}
_current++;
// Alloc _builder always 2 greater than locInBuilder to make sure
// we can do nested/escape parsing without error...
//
if (locInBuilder >= _builder.Length) {
char[] bigger = new char[_builder.Length + BufferGrowth];
Array.Copy(_builder, bigger, _builder.Length);
_builder = bigger;
_recyclable = false;
}
}
return new string(_builder, 0, locInBuilder);
}
private object ConsumeStringArray() {
ArrayList data = new ArrayList();
while (_deserializationData[_current] != rightAngleBracketChar) {
object itemValue = null;
if (_deserializationData[_current] != valueDelimiterChar) {
itemValue = ConsumeOneToken();
}
data.Add(itemValue);
_current++; //consume the delimiter
}
return data.ToArray(typeof(string));
}
private object ConsumeArray(Type creatableType) {
ArrayList data = new ArrayList();
while (_deserializationData[_current] != rightAngleBracketChar) {
object itemValue = null;
if (_deserializationData[_current] != valueDelimiterChar) {
itemValue = DeserializeValueInternal();
}
data.Add(itemValue);
_current++; //consume the delimiter
}
return data.ToArray(creatableType);
}
private object ConsumeTypeConverterValue(string token) {
int typeref = ParseNumericString(token);
TypeConverter tc;
if (typeref != -1) {
// token is the string representation of the number here
tc = (TypeConverter) _deserializedConverterTable[token];
if (tc == null) {
// wasn't in the converter table, add it now
// we need this case because arrays can add types but not typeconverters
Type t = TypeFromTypeCode(typeref);
tc = TypeDescriptor.GetConverter(t);
_deserializedConverterTable[token] = tc;
}
}
else {
// it's just a name, lookup type and add to type table
Type t = Type.GetType(token);
tc = TypeDescriptor.GetConverter(t);
// add to type table and converter table.
_deserializedConverterTable[(_deserializedTypeTable.Count + 50).ToString(NumberFormat)] = tc;
_deserializedTypeTable.Add(t);
}
string text = ConsumeOneToken();
return tc.ConvertFrom(null, CultureInfo.InvariantCulture, text);
}
/// <devdoc>
/// <para>Serializes the Web Forms view state value into
/// a <see cref='System.IO.Stream'/> object.</para>
/// </devdoc>
public void Serialize(Stream stream, object value) {
TextWriter output = new StreamWriter(stream);
SerializeInternal(output, value);
output.Flush();
}
/// <devdoc>
/// <para>Serializes the view state value into a <see cref='System.IO.TextWriter'/> object.</para>
/// </devdoc>
public void Serialize(TextWriter output, object value) {
SerializeInternal(output, value);
}
/// <devdoc>
/// Serialized value into the writer.
/// </devdoc>
private void SerializeInternal(TextWriter output, object value) {
if (value == null)
return;
if (_typeTable == null)
_typeTable = new HybridDictionary();
else
_typeTable.Clear();
#if NO_BASE64
SerializeValue(output, value);
#else
LosWriter writer = new LosWriter();
SerializeValue(writer, value);
writer.CompleteTransforms(output, EnableViewStateMac, _macKey);
writer.Dispose();
#endif
}
/// <devdoc>
/// Recursively serializes value into the writer.
/// </devdoc>
private void SerializeValue(TextWriter output, object value) {
if (value == null)
return;
// First determine the type... either typeless (string), array,
// typed array, hashtable, pair, triplet, knowntype, typetable reference, or
// type...
//
// serialize string...
//
if (value is string) {
WriteEscapedString(output, (string)value);
}
// serialize Int32...
//
else if (value is Int32) {
output.Write('i');
output.Write(leftAngleBracketChar);
output.Write(((Int32)value).ToString(NumberFormat));
output.Write(rightAngleBracketChar);
}
else if (value is Boolean) {
output.Write('o');
output.Write(leftAngleBracketChar);
output.Write( ((bool) value) ? 't' : 'f');
output.Write(rightAngleBracketChar);
}
// serialize arraylist...
//
else if (value is ArrayList) {
output.Write('l');
output.Write(leftAngleBracketChar);
ArrayList ar = (ArrayList)value;
int c = ar.Count;
for (int i=0; i<c; i++) {
SerializeValue(output, ar[i]);
output.Write(valueDelimiterChar);
}
output.Write(rightAngleBracketChar);
}
// serialize hashtable...
//
else if (value is Hashtable) {
output.Write('h');
output.Write(leftAngleBracketChar);
Hashtable table = (Hashtable)value;
IDictionaryEnumerator e = table.GetEnumerator();
while (e.MoveNext()) {
SerializeValue(output, e.Key);
output.Write(valueDelimiterChar);
SerializeValue(output, e.Value);
output.Write(valueDelimiterChar);
}
output.Write(rightAngleBracketChar);
}
else {
// we'll need the Type object for the last two possibilities
Type valueType = value.GetType();
Type strtype = typeof(string);
// serialize Pair
if (valueType == typeof(Pair)) {
Pair p = (Pair) value;
output.Write('p');
output.Write(leftAngleBracketChar);
SerializeValue(output, p.First);
output.Write(valueDelimiterChar);
SerializeValue(output, p.Second);
output.Write(rightAngleBracketChar);
}
// serialize Triplet
//
else if (valueType == typeof(Triplet)) {
Triplet t = (Triplet) value;
output.Write('t');
output.Write(leftAngleBracketChar);
SerializeValue(output, t.First);
output.Write(valueDelimiterChar);
SerializeValue(output, t.Second);
output.Write(valueDelimiterChar);
SerializeValue(output, t.Third);
output.Write(rightAngleBracketChar);
}
// serialize array...
//
else if (valueType.IsArray) {
Type underlyingValueType;
underlyingValueType = valueType.GetElementType();
output.Write('@');
if (underlyingValueType != strtype) {
// write type of array before elements
int typeId = GetTypeId(underlyingValueType);
WriteTypeId(output, typeId, underlyingValueType);
output.Write(leftAngleBracketChar);
Array ar = (Array)value;
for (int i=0; i<ar.Length; i++) {
SerializeValue(output, ar.GetValue(i));
output.Write(valueDelimiterChar);
}
}
else {
// optimization: since we know the underlying values are strings,
// we can skip the recursive call to SerializeValue
output.Write(leftAngleBracketChar);
string[] ar = (string[])value;
for (int i=0; i<ar.Length; i++) {
WriteEscapedString(output, ar[i]);
output.Write(valueDelimiterChar);
}
}
output.Write(rightAngleBracketChar);
}
// serialize other value...
//
else {
int typeId = GetTypeId(valueType);
// get the type converter
TypeConverter tc = TypeDescriptor.GetConverter(valueType);
bool toString;
bool fromString;
if (tc == null || tc is ReferenceConverter) {
toString = false;
fromString = false;
}
else {
toString = tc.CanConvertTo(strtype);
fromString = tc.CanConvertFrom(strtype);
}
if (toString && fromString) {
//we can convert to and from a string
WriteTypeId(output, typeId, valueType);
output.Write(leftAngleBracketChar);
WriteEscapedString(output, tc.ConvertToInvariantString(null, value));
}
else {
// the typeconverter failed us, so we are resorting to binary serialization
MemoryStream ms = new MemoryStream();
System.Runtime.Serialization.IFormatter formatter = new System.Runtime.Serialization.Formatters.Binary.BinaryFormatter();
try {
formatter.Serialize(ms, value);
}
catch (SerializationException) {
throw new HttpException(SR.GetString(SR.NonSerializableType, value.GetType().FullName));
}
output.Write('b');
output.Write(leftAngleBracketChar);
// Since base64 doesn't have any chars that we escape, we can
// skip WriteEscapedString
output.Write(Convert.ToBase64String(ms.GetBuffer(), 0, (int) ms.Length));
}
output.Write(rightAngleBracketChar);
}
}
}
private int GetTypeId(Type valueType) {
int typeId = NoTypeId;
// check if it is a known type
for (int i=0; i<knownTypes.Length; i++) {
if (valueType == knownTypes[i]) {
typeId = i;
break;
}
}
if (typeId == NoTypeId) {
// not a known type, see if it's in the type table
object found = _typeTable[valueType];
if (found != null)
typeId = 50 + (int)found;
}
return typeId;
}
private void WriteTypeId(TextWriter output, int typeId, Type valueType) {
if (typeId != NoTypeId)
output.Write(typeId.ToString(NumberFormat));
else {
// ASURT 60173: use AssemblyQualifiedName here, not FullName
WriteEscapedString(output, valueType.AssemblyQualifiedName);
//
_typeTable[valueType] = _typeTable.Count;
}
}
/// <devdoc>
/// Takes a typeRef, and converts it to a Type. Either by returning
/// Type.GetType(typeRef), or looking it up.
/// </devdoc>
private Type TypeFromTypeRef(string typeRef) {
int number = ParseNumericString(typeRef);
Type t = TypeFromTypeCode(number);
if (t != null)
return t;
// it's just a name, lookup type and add to type table
t = Type.GetType(typeRef);
_deserializedTypeTable.Add(t);
return t;
}
private Type TypeFromTypeCode(int number) {
if (number != -1) {
// it is a type id, either in the known table or in our type table
if (number <= 49)
return knownTypes[number];
return (Type) _deserializedTypeTable[number - 50];
}
return null;
}
// Note : We have to determine if "typeRef" is a number. The easiest
// : and fastest way to do this is to walk the string. While
// : we are doing this, lets build up the number... after
// : all this is much faster than Int32.Parse
//
private int ParseNumericString(string num) {
int number = 0;
int len = num.Length;
for (int i=0; i<len; i++) {
switch (num[i]) {
case '0':
case '1':
case '2':
case '3':
case '4':
case '5':
case '6':
case '7':
case '8':
case '9':
number *= 10;
number += (((int)num[i]) - ((int)'0'));
break;
default:
number = -1;
i = len;
break;
}
}
return number;
}
/// <devdoc>
/// Escapes and writes the escaped value of str into the writer.
/// </devdoc>
private void WriteEscapedString(TextWriter output, string str) {
if (str == null)
return;
// need to "escape" the empty string to distinguish it
// from a null value
if (str.Length == 0) {
output.Write('\\');
output.Write('e');
return;
}
int first = str.IndexOfAny(escapedCharacters);
if (first == -1) {
output.Write(str);
}
else {
char[] strData = str.ToCharArray();
output.Write(strData, 0, first);
int len = strData.Length;
for (int i=first; i<len; i++) {
char c = strData[i];
switch (c) {
case '\\':
output.Write('\\');
output.Write('\\');
break;
case leftAngleBracketChar:
output.Write('\\');
output.Write(leftAngleBracketChar);
break;
case rightAngleBracketChar:
output.Write('\\');
output.Write(rightAngleBracketChar);
break;
case valueDelimiterChar:
output.Write('\\');
output.Write(valueDelimiterChar);
break;
default:
output.Write(c);
break;
}
}
}
}
static internal int EstimateSize(object obj) {
if (obj == null)
return 0;
StringWriter sw = new StringWriter();
LosFormatter formatter = new LosFormatter();
formatter.Serialize(sw, obj);
return sw.ToString().Length;
}
#region Implementation of IStateFormatter
object IStateFormatter.Deserialize(string serializedState) {
return Deserialize(serializedState);
}
string IStateFormatter.Serialize(object state) {
StringWriter writer = new StringWriter();
Serialize(writer, state);
return writer.ToString();
}
#endregion
}
}
#endif // OBJECTSTATEFORMATTER