// 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. // // Copyright (c) 2004-2005 Novell, Inc. // // Authors: // Duncan Mak duncan@ximian.com // Gonzalo Paniagua Javier gonzalo@ximian.com // Peter Bartok pbartok@novell.com // // COMPLETE using System.ComponentModel; using System.IO; using System.Runtime.Serialization.Formatters.Binary; using System.Text; using System.Xml; using System.Reflection; namespace System.Resources { #if INSIDE_SYSTEM_WEB internal #else public #endif class ResXResourceWriter : IResourceWriter, IDisposable { #region Local Variables private string filename; private Stream stream; private TextWriter textwriter; private XmlTextWriter writer; private bool written; #if NET_2_0 private string base_path; #endif #endregion // Local Variables #region Static Fields public static readonly string BinSerializedObjectMimeType = "application/x-microsoft.net.object.binary.base64"; public static readonly string ByteArraySerializedObjectMimeType = "application/x-microsoft.net.object.bytearray.base64"; public static readonly string DefaultSerializedObjectMimeType = BinSerializedObjectMimeType; public static readonly string ResMimeType = "text/microsoft-resx"; public static readonly string ResourceSchema = schema; public static readonly string SoapSerializedObjectMimeType = "application/x-microsoft.net.object.soap.base64"; #if NET_2_0 public static readonly string Version = "2.0"; #else public static readonly string Version = "1.3"; #endif #endregion // Static Fields #region Constructors & Destructor public ResXResourceWriter (Stream stream) { if (stream == null) throw new ArgumentNullException ("stream"); if (!stream.CanWrite) throw new ArgumentException ("stream is not writable.", "stream"); this.stream = stream; } public ResXResourceWriter (TextWriter textWriter) { if (textWriter == null) throw new ArgumentNullException ("textWriter"); this.textwriter = textWriter; } public ResXResourceWriter (string fileName) { if (fileName == null) throw new ArgumentNullException ("fileName"); this.filename = fileName; } ~ResXResourceWriter() { Dispose(false); } #endregion // Constructors & Destructor void InitWriter () { if (filename != null) stream = File.OpenWrite (filename); if (textwriter == null) textwriter = new StreamWriter (stream, Encoding.UTF8); writer = new XmlTextWriter (textwriter); writer.Formatting = Formatting.Indented; writer.WriteStartDocument (); writer.WriteStartElement ("root"); writer.WriteRaw (schema); WriteHeader ("resmimetype", "text/microsoft-resx"); WriteHeader ("version", "1.3"); WriteHeader ("reader", typeof (ResXResourceReader).AssemblyQualifiedName); WriteHeader ("writer", typeof (ResXResourceWriter).AssemblyQualifiedName); } void WriteHeader (string name, string value) { writer.WriteStartElement ("resheader"); writer.WriteAttributeString ("name", name); writer.WriteStartElement ("value"); writer.WriteString (value); writer.WriteEndElement (); writer.WriteEndElement (); } void WriteNiceBase64(byte[] value, int offset, int length) { string b64; StringBuilder sb; int pos; int inc; string ins; b64 = Convert.ToBase64String(value, offset, length); // Wild guess; two extra newlines, and one newline/tab pair for every 80 chars sb = new StringBuilder(b64, b64.Length + ((b64.Length + 160) / 80) * 3); pos = 0; inc = 80 + Environment.NewLine.Length + 1; ins = Environment.NewLine + "\t"; while (pos < sb.Length) { sb.Insert(pos, ins); pos += inc; } sb.Insert(sb.Length, Environment.NewLine); writer.WriteString(sb.ToString()); } void WriteBytes (string name, Type type, byte[] value, int offset, int length) { WriteBytes (name, type, value, offset, length, String.Empty); } void WriteBytes (string name, Type type, byte[] value, int offset, int length, string comment) { writer.WriteStartElement ("data"); writer.WriteAttributeString ("name", name); if (type != null) { writer.WriteAttributeString ("type", type.AssemblyQualifiedName); // byte[] should never get a mimetype, otherwise MS.NET won't be able // to parse the data. if (type != typeof (byte[])) writer.WriteAttributeString ("mimetype", ByteArraySerializedObjectMimeType); writer.WriteStartElement ("value"); WriteNiceBase64 (value, offset, length); } else { writer.WriteAttributeString ("mimetype", BinSerializedObjectMimeType); writer.WriteStartElement ("value"); writer.WriteBase64 (value, offset, length); } writer.WriteEndElement (); if (!(comment == null || comment.Equals (String.Empty))) { writer.WriteStartElement ("comment"); writer.WriteString (comment); writer.WriteEndElement (); } writer.WriteEndElement (); } void WriteBytes (string name, Type type, byte [] value) { WriteBytes (name, type, value, 0, value.Length); } void WriteString (string name, string value) { WriteString (name, value, null); } void WriteString (string name, string value, Type type) { WriteString (name, value, type, String.Empty); } void WriteString (string name, string value, Type type, string comment) { writer.WriteStartElement ("data"); writer.WriteAttributeString ("name", name); if (type != null) writer.WriteAttributeString ("type", type.AssemblyQualifiedName); writer.WriteStartElement ("value"); writer.WriteString (value); writer.WriteEndElement (); if (!(comment == null || comment.Equals (String.Empty))) { writer.WriteStartElement ("comment"); writer.WriteString (comment); writer.WriteEndElement (); } writer.WriteEndElement (); writer.WriteWhitespace ("\n "); } public void AddResource (string name, byte [] value) { if (name == null) throw new ArgumentNullException ("name"); if (value == null) throw new ArgumentNullException ("value"); if (written) throw new InvalidOperationException ("The resource is already generated."); if (writer == null) InitWriter (); WriteBytes (name, value.GetType (), value); } public void AddResource (string name, object value) { AddResource (name, value, String.Empty); } private void AddResource (string name, object value, string comment) { if (value is string) { AddResource (name, (string) value, comment); return; } if (value is byte[]) { AddResource (name, (byte[]) value); return; } if (name == null) throw new ArgumentNullException ("name"); if (value == null) throw new ArgumentNullException ("value"); if (!value.GetType ().IsSerializable) throw new InvalidOperationException (String.Format ("The element '{0}' of type '{1}' is not serializable.", name, value.GetType ().Name)); if (written) throw new InvalidOperationException ("The resource is already generated."); if (writer == null) InitWriter (); TypeConverter converter = TypeDescriptor.GetConverter (value); if (converter != null && converter.CanConvertTo (typeof (string)) && converter.CanConvertFrom (typeof (string))) { string str = (string) converter.ConvertToInvariantString (value); WriteString (name, str, value.GetType ()); return; } if (converter != null && converter.CanConvertTo (typeof (byte[])) && converter.CanConvertFrom (typeof (byte[]))) { byte[] b = (byte[]) converter.ConvertTo (value, typeof (byte[])); WriteBytes (name, value.GetType (), b); return; } MemoryStream ms = new MemoryStream (); BinaryFormatter fmt = new BinaryFormatter (); try { fmt.Serialize (ms, value); } catch (Exception e) { throw new InvalidOperationException ("Cannot add a " + value.GetType () + "because it cannot be serialized: " + e.Message); } WriteBytes (name, null, ms.GetBuffer (), 0, (int) ms.Length, comment); ms.Close (); } public void AddResource (string name, string value) { AddResource (name, value, string.Empty); } private void AddResource (string name, string value, string comment) { if (name == null) throw new ArgumentNullException ("name"); if (value == null) throw new ArgumentNullException ("value"); if (written) throw new InvalidOperationException ("The resource is already generated."); if (writer == null) InitWriter (); WriteString (name, value, null, comment); } #if NET_2_0 [MonoTODO ("Stub, not implemented")] public virtual void AddAlias (string aliasName, AssemblyName assemblyName) { } public void AddResource (ResXDataNode node) { AddResource (node.Name, node.Value, node.Comment); } public void AddMetadata (string name, string value) { if (name == null) throw new ArgumentNullException ("name"); if (value == null) throw new ArgumentNullException ("value"); if (written) throw new InvalidOperationException ("The resource is already generated."); if (writer == null) InitWriter (); writer.WriteStartElement ("metadata"); writer.WriteAttributeString ("name", name); writer.WriteAttributeString ("xml:space", "preserve"); writer.WriteElementString ("value", value); writer.WriteEndElement (); } public void AddMetadata (string name, byte[] value) { if (name == null) throw new ArgumentNullException ("name"); if (value == null) throw new ArgumentNullException ("value"); if (written) throw new InvalidOperationException ("The resource is already generated."); if (writer == null) InitWriter (); writer.WriteStartElement ("metadata"); writer.WriteAttributeString ("name", name); writer.WriteAttributeString ("type", value.GetType ().AssemblyQualifiedName); writer.WriteStartElement ("value"); WriteNiceBase64 (value, 0, value.Length); writer.WriteEndElement (); writer.WriteEndElement (); } public void AddMetadata (string name, object value) { if (value is string) { AddMetadata (name, (string)value); return; } if (value is byte[]) { AddMetadata (name, (byte[])value); return; } if (name == null) throw new ArgumentNullException ("name"); if (value == null) throw new ArgumentNullException ("value"); if (!value.GetType ().IsSerializable) throw new InvalidOperationException (String.Format ("The element '{0}' of type '{1}' is not serializable.", name, value.GetType ().Name)); if (written) throw new InvalidOperationException ("The resource is already generated."); if (writer == null) InitWriter (); Type type = value.GetType (); TypeConverter converter = TypeDescriptor.GetConverter (value); if (converter != null && converter.CanConvertTo (typeof (string)) && converter.CanConvertFrom (typeof (string))) { string str = (string)converter.ConvertToInvariantString (value); writer.WriteStartElement ("metadata"); writer.WriteAttributeString ("name", name); if (type != null) writer.WriteAttributeString ("type", type.AssemblyQualifiedName); writer.WriteStartElement ("value"); writer.WriteString (str); writer.WriteEndElement (); writer.WriteEndElement (); writer.WriteWhitespace ("\n "); return; } if (converter != null && converter.CanConvertTo (typeof (byte[])) && converter.CanConvertFrom (typeof (byte[]))) { byte[] b = (byte[])converter.ConvertTo (value, typeof (byte[])); writer.WriteStartElement ("metadata"); writer.WriteAttributeString ("name", name); if (type != null) { writer.WriteAttributeString ("type", type.AssemblyQualifiedName); writer.WriteAttributeString ("mimetype", ByteArraySerializedObjectMimeType); writer.WriteStartElement ("value"); WriteNiceBase64 (b, 0, b.Length); } else { writer.WriteAttributeString ("mimetype", BinSerializedObjectMimeType); writer.WriteStartElement ("value"); writer.WriteBase64 (b, 0, b.Length); } writer.WriteEndElement (); writer.WriteEndElement (); return; } MemoryStream ms = new MemoryStream (); BinaryFormatter fmt = new BinaryFormatter (); try { fmt.Serialize (ms, value); } catch (Exception e) { throw new InvalidOperationException ("Cannot add a " + value.GetType () + "because it cannot be serialized: " + e.Message); } writer.WriteStartElement ("metadata"); writer.WriteAttributeString ("name", name); if (type != null) { writer.WriteAttributeString ("type", type.AssemblyQualifiedName); writer.WriteAttributeString ("mimetype", ByteArraySerializedObjectMimeType); writer.WriteStartElement ("value"); WriteNiceBase64 (ms.GetBuffer (), 0, ms.GetBuffer ().Length); } else { writer.WriteAttributeString ("mimetype", BinSerializedObjectMimeType); writer.WriteStartElement ("value"); writer.WriteBase64 (ms.GetBuffer (), 0, ms.GetBuffer ().Length); } writer.WriteEndElement (); writer.WriteEndElement (); ms.Close (); } #endif public void Close () { if (!written) { Generate (); } if (writer != null) { writer.Close (); stream = null; filename = null; textwriter = null; } } public virtual void Dispose () { Dispose(true); GC.SuppressFinalize(this); } public void Generate () { if (written) throw new InvalidOperationException ("The resource is already generated."); written = true; writer.WriteEndElement (); writer.Flush (); } protected virtual void Dispose (bool disposing) { if (disposing) Close(); } static string schema = @" ".Replace ("'", "\""); #region Public Properties #if NET_2_0 public string BasePath { get { return base_path; } set { base_path = value; } } #endif #endregion } }