Trigger4Win/libcmdline/Infrastructure/ReflectionHelper.cs
2015-04-10 00:09:58 +00:00

248 lines
9.2 KiB
C#

#region License
// <copyright file="ReflectionHelper.cs" company="Giacomo Stelluti Scala">
// Copyright 2015-2013 Giacomo Stelluti Scala
// </copyright>
//
// 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.
#endregion
#region Using Directives
using System;
using System.Collections.Generic;
using System.Reflection;
#endregion
namespace CommandLine.Infrastructure
{
internal static class ReflectionHelper
{
static ReflectionHelper()
{
AssemblyFromWhichToPullInformation = Assembly.GetEntryAssembly() ?? Assembly.GetExecutingAssembly();
}
/// <summary>
/// Gets or sets the assembly from which to pull information. Setter provided for testing purpose.
/// </summary>
internal static Assembly AssemblyFromWhichToPullInformation
{
get; set;
}
public static IList<Pair<PropertyInfo, TAttribute>> RetrievePropertyList<TAttribute>(object target)
where TAttribute : Attribute
{
var key = new Pair<Type, object>(typeof(Pair<PropertyInfo, TAttribute>), target);
var cached = ReflectionCache.Instance[key];
if (cached == null)
{
IList<Pair<PropertyInfo, TAttribute>> list = new List<Pair<PropertyInfo, TAttribute>>();
if (target != null)
{
var propertiesInfo = target.GetType().GetProperties();
foreach (var property in propertiesInfo)
{
if (property == null || (!property.CanRead || !property.CanWrite))
{
continue;
}
var setMethod = property.GetSetMethod();
if (setMethod == null || setMethod.IsStatic)
{
continue;
}
var attribute = Attribute.GetCustomAttribute(property, typeof(TAttribute), false);
if (attribute != null)
{
list.Add(new Pair<PropertyInfo, TAttribute>(property, (TAttribute)attribute));
}
}
}
ReflectionCache.Instance[key] = list;
return list;
}
return (IList<Pair<PropertyInfo, TAttribute>>)cached;
}
public static Pair<MethodInfo, TAttribute> RetrieveMethod<TAttribute>(object target)
where TAttribute : Attribute
{
var key = new Pair<Type, object>(typeof(Pair<MethodInfo, TAttribute>), target);
var cached = ReflectionCache.Instance[key];
if (cached == null)
{
var info = target.GetType().GetMethods();
foreach (var method in info)
{
if (method.IsStatic)
{
continue;
}
var attribute = Attribute.GetCustomAttribute(method, typeof(TAttribute), false);
if (attribute == null)
{
continue;
}
var data = new Pair<MethodInfo, TAttribute>(method, (TAttribute)attribute);
ReflectionCache.Instance[key] = data;
return data;
}
return null;
}
return (Pair<MethodInfo, TAttribute>)cached;
}
public static TAttribute RetrieveMethodAttributeOnly<TAttribute>(object target)
where TAttribute : Attribute
{
var key = new Pair<Type, object>(typeof(TAttribute), target);
var cached = ReflectionCache.Instance[key];
if (cached == null)
{
var info = target.GetType().GetMethods();
foreach (var method in info)
{
if (method.IsStatic)
{
continue;
}
var attribute = Attribute.GetCustomAttribute(method, typeof(TAttribute), false);
if (attribute == null)
{
continue;
}
var data = (TAttribute)attribute;
ReflectionCache.Instance[key] = data;
return data;
}
return null;
}
return (TAttribute)cached;
}
public static IList<TAttribute> RetrievePropertyAttributeList<TAttribute>(object target)
where TAttribute : Attribute
{
var key = new Pair<Type, object>(typeof(IList<TAttribute>), target);
var cached = ReflectionCache.Instance[key];
if (cached == null)
{
IList<TAttribute> list = new List<TAttribute>();
var info = target.GetType().GetProperties();
foreach (var property in info)
{
if (property == null || (!property.CanRead || !property.CanWrite))
{
continue;
}
var setMethod = property.GetSetMethod();
if (setMethod == null || setMethod.IsStatic)
{
continue;
}
var attribute = Attribute.GetCustomAttribute(property, typeof(TAttribute), false);
if (attribute != null)
{
list.Add((TAttribute)attribute);
}
}
ReflectionCache.Instance[key] = list;
return list;
}
return (IList<TAttribute>)cached;
}
public static TAttribute GetAttribute<TAttribute>()
where TAttribute : Attribute
{
var a = AssemblyFromWhichToPullInformation.GetCustomAttributes(typeof(TAttribute), false);
if (a.Length <= 0)
{
return null;
}
return (TAttribute)a[0];
}
public static Pair<PropertyInfo, TAttribute> RetrieveOptionProperty<TAttribute>(object target, string uniqueName)
where TAttribute : BaseOptionAttribute
{
var key = new Pair<Type, object>(typeof(Pair<PropertyInfo, BaseOptionAttribute>), target);
var cached = ReflectionCache.Instance[key];
if (cached == null)
{
if (target == null)
{
return null;
}
var propertiesInfo = target.GetType().GetProperties();
foreach (var property in propertiesInfo)
{
if (property == null || (!property.CanRead || !property.CanWrite))
{
continue;
}
var setMethod = property.GetSetMethod();
if (setMethod == null || setMethod.IsStatic)
{
continue;
}
var attribute = Attribute.GetCustomAttribute(property, typeof(TAttribute), false);
var optionAttr = (TAttribute)attribute;
if (optionAttr == null || string.CompareOrdinal(uniqueName, optionAttr.UniqueName) != 0)
{
continue;
}
var found = new Pair<PropertyInfo, TAttribute>(property, (TAttribute)attribute);
ReflectionCache.Instance[key] = found;
return found;
}
}
return (Pair<PropertyInfo, TAttribute>)cached;
}
public static bool IsNullableType(Type type)
{
return type.IsGenericType && type.GetGenericTypeDefinition() == typeof(Nullable<>);
}
}
}