using System; using System.Collections.Generic; using System.Text; using System.Collections; using System.IO; using System.Reflection; namespace Google.ProtocolBuffers { /// /// Helper class to create MessageStreamIterators without explicitly specifying /// the builder type. The concrete builder type is determined by reflection, based /// on the message type. The reflection step looks for a CreateBuilder method /// in the message type which is public and static, and uses the return type of that /// method as the builder type. This will work for all generated messages, whether /// optimised for size or speed. It won't work for dynamic messages. /// /// TODO(jonskeet): This also won't work for non-public protos :( /// Pass in delegate to create a builder? /// public static class MessageStreamIterator { public static IEnumerable FromFile(string file) where TMessage : IMessage { return FromStreamProvider(() => File.OpenRead(file)); } public static IEnumerable FromStreamProvider(StreamProvider streamProvider) where TMessage : IMessage { MethodInfo createBuilderMethod = typeof(TMessage).GetMethod("CreateBuilder", Type.EmptyTypes); if (createBuilderMethod == null) { throw new ArgumentException("Message type " + typeof(TMessage).FullName + " has no CreateBuilder method."); } if (createBuilderMethod.ReturnType == typeof(void)) { throw new ArgumentException("CreateBuilder method in " + typeof(TMessage).FullName + " has void return type"); } Type builderType = createBuilderMethod.ReturnType; if (builderType.GetConstructor(Type.EmptyTypes) == null) { throw new ArgumentException("Builder type " + builderType.FullName + " has no public parameterless constructor."); } Type messageInterface = typeof(IMessage<,>).MakeGenericType(typeof(TMessage), builderType); Type builderInterface = typeof(IBuilder<,>).MakeGenericType(typeof(TMessage), builderType); if (Array.IndexOf(typeof (TMessage).GetInterfaces(), messageInterface) == -1) { throw new ArgumentException("Message type " + typeof(TMessage) + " doesn't implement " + messageInterface.FullName); } if (Array.IndexOf(builderType.GetInterfaces(), builderInterface) == -1) { throw new ArgumentException("Builder type " + typeof(TMessage) + " doesn't implement " + builderInterface.FullName); } Type iteratorType = typeof(MessageStreamIterator<,>).MakeGenericType(typeof(TMessage), builderType); MethodInfo factoryMethod = iteratorType.GetMethod("FromStreamProvider", new Type[] { typeof(StreamProvider) }); return (IEnumerable) factoryMethod.Invoke(null, new object[] { streamProvider }); } } /// /// Iterates over data created using a . /// Unlike MessageStreamWriter, this class is not usually constructed directly with /// a stream; instead it is provided with a way of opening a stream when iteration /// is started. The stream is closed when the iteration is completed or the enumerator /// is disposed. (This occurs naturally when using foreach.) /// This type is generic in both the message type and the builder type; if only the /// iteration is required (i.e. just ) then the static /// generic methods in the nongeneric class are more appropriate. /// public sealed class MessageStreamIterator : IEnumerable where TMessage : IMessage where TBuilder : IBuilder, new() { private readonly StreamProvider streamProvider; private readonly ExtensionRegistry extensionRegistry; private static readonly uint ExpectedTag = WireFormat.MakeTag(1, WireFormat.WireType.LengthDelimited); private MessageStreamIterator(StreamProvider streamProvider, ExtensionRegistry extensionRegistry) { this.streamProvider = streamProvider; this.extensionRegistry = extensionRegistry; } /// /// Creates an instance which opens the specified file when it begins /// iterating. No extension registry is used when reading messages. /// public static MessageStreamIterator FromFile(string file) { return new MessageStreamIterator(() => File.OpenRead(file), ExtensionRegistry.Empty); } /// /// Creates an instance which calls the given delegate when it begins /// iterating. No extension registry is used when reading messages. /// public static MessageStreamIterator FromStreamProvider(StreamProvider streamProvider) { return new MessageStreamIterator(streamProvider, ExtensionRegistry.Empty); } /// /// Creates a new instance which uses the same stream provider as this one, /// but the specified extension registry. /// public MessageStreamIterator WithExtensionRegistry(ExtensionRegistry newRegistry) { return new MessageStreamIterator(streamProvider, newRegistry); } public IEnumerator GetEnumerator() { using (Stream stream = streamProvider()) { CodedInputStream input = CodedInputStream.CreateInstance(stream); uint tag; while ((tag = input.ReadTag()) != 0) { if (tag != ExpectedTag) { throw InvalidProtocolBufferException.InvalidMessageStreamTag(); } TBuilder builder = new TBuilder(); input.ReadMessage(builder, extensionRegistry); yield return builder.Build(); } } } /// /// Explicit implementation of nongeneric IEnumerable interface. /// IEnumerator IEnumerable.GetEnumerator() { return GetEnumerator(); } } }