// // System.Resources.ResourceReader.cs // // Authors: // Duncan Mak // Nick Drochak // Dick Porter // Marek Safar // Atsushi Enomoto // // (C) 2001, 2002 Ximian Inc, http://www.ximian.com // Copyright (C) 2004-2005,2007 Novell, Inc (http://www.novell.com) // // 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.Collections; using System.Resources; using System.IO; using System.Text; using System.Runtime.InteropServices; using System.Runtime.Serialization; using System.Runtime.Serialization.Formatters.Binary; using System.Security.Permissions; namespace System.Resources { internal enum PredefinedResourceType { Null = 0, String = 1, Bool = 2, Char = 3, Byte = 4, SByte = 5, Int16 = 6, UInt16 = 7, Int32 = 8, UInt32 = 9, Int64 = 10, UInt64 = 11, Single = 12, Double = 13, Decimal = 14, DateTime = 15, TimeSpan = 16, ByteArray = 32, Stream = 33, FistCustom = 64 } #if NET_2_0 [System.Runtime.InteropServices.ComVisible (true)] #endif public sealed class ResourceReader : IResourceReader, IEnumerable, IDisposable { BinaryReader reader; IFormatter formatter; internal int resourceCount = 0; int typeCount = 0; string[] typeNames; int[] hashes; long[] positions; int dataSectionOffset; long nameSectionOffset; int resource_ver; // Constructors [SecurityPermission (SecurityAction.LinkDemand, SerializationFormatter = true)] public ResourceReader (Stream stream) { if (stream == null) throw new ArgumentNullException ("stream"); if (!stream.CanRead) throw new ArgumentException ("Stream was not readable."); reader = new BinaryReader(stream, Encoding.UTF8); formatter = new BinaryFormatter(null, new StreamingContext(StreamingContextStates.File|StreamingContextStates.Persistence)); ReadHeaders(); } public ResourceReader (string fileName) { reader = new BinaryReader (new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read)); formatter = new BinaryFormatter(null, new StreamingContext(StreamingContextStates.File|StreamingContextStates.Persistence)); ReadHeaders(); } /* Read the ResourceManager header and the * ResourceReader header. */ private void ReadHeaders() { try { int manager_magic = reader.ReadInt32(); if(manager_magic != ResourceManager.MagicNumber) throw new ArgumentException(String.Format ("Stream is not a valid .resources file, magic=0x{0:x}", manager_magic)); int manager_ver = reader.ReadInt32(); int manager_len = reader.ReadInt32(); /* We know how long the header is, even if * the version number is too new */ if(manager_ver > ResourceManager.HeaderVersionNumber) { reader.BaseStream.Seek(manager_len, SeekOrigin.Current); } else { string reader_class=reader.ReadString(); if(!reader_class.StartsWith("System.Resources.ResourceReader")) { throw new NotSupportedException("This .resources file requires reader class " + reader_class); } string set_class=reader.ReadString(); if(!set_class.StartsWith(typeof(ResourceSet).FullName) && !set_class.StartsWith("System.Resources.RuntimeResourceSet")) { throw new NotSupportedException("This .resources file requires set class " + set_class); } } /* Now read the ResourceReader header */ resource_ver = reader.ReadInt32(); if(resource_ver != 1 #if NET_2_0 && resource_ver != 2 #endif ) { throw new NotSupportedException("This .resources file requires unsupported set class version: " + resource_ver.ToString()); } resourceCount = reader.ReadInt32(); typeCount = reader.ReadInt32(); typeNames=new string[typeCount]; for(int i=0; i= (int) PredefinedResourceType.FistCustom) { int typenameidx = type_index - (int)PredefinedResourceType.FistCustom; if (typenameidx >= typeNames.Length) throw new FormatException ("The resource data is corrupt. Invalid index to types"); resourceType = typeNames[typenameidx]; } else resourceType = "ResourceTypeCode." + (PredefinedResourceType) type_index; ReadValueVer2 (type_index); } else { // resource ver 1 == untyped resourceType = "ResourceTypeCode.Null"; ReadValueVer1 (Type.GetType (typeNames [type_index], true)); } // FIXME: the data size is wrong. int datalen = (int) (reader.BaseStream.Position - pos2); reader.BaseStream.Seek (-datalen, SeekOrigin.Current); data = new byte [datalen]; reader.BaseStream.Read (data, 0, datalen); } } #endif void IDisposable.Dispose () { Dispose(true); } private void Dispose (bool disposing) { if(disposing) { if(reader!=null) { reader.Close(); } } reader=null; hashes=null; positions=null; typeNames=null; } internal class ResourceEnumerator : IDictionaryEnumerator { private ResourceReader reader; private int index = -1; private bool finished = false; internal ResourceEnumerator(ResourceReader readerToEnumerate){ reader = readerToEnumerate; } public int Index { get { return index; } } public virtual DictionaryEntry Entry { get { if (reader.reader == null) throw new InvalidOperationException("ResourceReader is closed."); if (index < 0) throw new InvalidOperationException("Enumeration has not started. Call MoveNext."); DictionaryEntry entry = new DictionaryEntry(); entry.Key = Key; entry.Value = Value; return entry; } } public virtual object Key { get { if (reader.reader == null) throw new InvalidOperationException("ResourceReader is closed."); if (index < 0) throw new InvalidOperationException("Enumeration has not started. Call MoveNext."); return (reader.ResourceName(index)); } } public virtual object Value { get { if (reader.reader == null) throw new InvalidOperationException("ResourceReader is closed."); if (index < 0) throw new InvalidOperationException("Enumeration has not started. Call MoveNext."); return(reader.ResourceValue(index)); } } #if NET_2_0 public UnmanagedMemoryStream ValueAsStream { get { if (reader.reader == null) throw new InvalidOperationException("ResourceReader is closed."); if (index < 0) throw new InvalidOperationException("Enumeration has not started. Call MoveNext."); return(reader.ResourceValueAsStream((string) Key, index)); } } #endif public virtual object Current { get { /* Entry does the checking, no * need to repeat it here */ return Entry; } } public virtual bool MoveNext () { if (reader.reader == null) throw new InvalidOperationException("ResourceReader is closed."); if (finished) { return false; } if (++index < reader.resourceCount){ return true; } else { finished=true; return false; } } public void Reset () { if (reader.reader == null) throw new InvalidOperationException("ResourceReader is closed."); index = -1; finished = false; } } // internal class ResourceEnumerator } // public sealed class ResourceReader } // namespace System.Resources