using System; using System.Collections; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Text; using System.Xml; using System.Xml.Schema; using System.IO; using System.Reflection; using System.Diagnostics; using System.Runtime.Serialization; using System.Security.Permissions; using System.Globalization; //using System.Workflow.Activities; using System.Workflow.ComponentModel; using System.Workflow.Runtime; using System.Workflow.Runtime.Hosting; using Hosting = System.Workflow.Runtime.Hosting; namespace System.Workflow.Runtime.Tracking { internal sealed class PropertyHelper { private PropertyHelper() { } #region Internal Static Methods internal static void GetProperty(string name, Activity activity, TrackingAnnotationCollection annotations, out TrackingDataItem item) { item = null; object tmp = PropertyHelper.GetProperty(name, activity); item = new TrackingDataItem(); item.FieldName = name; item.Data = tmp; foreach (string s in annotations) item.Annotations.Add(s); } internal static object GetProperty(string name, object obj) { if (null == name) throw new ArgumentNullException("name"); if (null == obj) throw new ArgumentNullException("obj"); // // Split the names string[] names = name.Split(new char[] { '.' }); object currObj = obj; for (int i = 0; i < names.Length; i++) { if ((null == names[i]) || (0 == names[i].Length)) throw new InvalidOperationException(ExecutionStringManager.TrackingProfileInvalidMember); object tmp = null; PropertyHelper.GetPropertyOrField(names[i], currObj, out tmp); // // Attempt to resolve runtime values (ParameterBinding, ParameterDeclaration and Bind) if (currObj is Activity) currObj = GetRuntimeValue(tmp, (Activity)currObj); else currObj = tmp; } return currObj; } internal static void GetPropertyOrField(string name, object o, out object obj) { obj = null; if (null == name) throw new ArgumentNullException("name"); if (null == o) throw new ArgumentNullException("o"); Type t = o.GetType(); string tmp = null, realName = null; bool isCollection = false; int index = -1; if (TryParseIndex(name, out tmp, out index)) isCollection = true; else tmp = name; object val = null; if ((null != tmp) && (tmp.Length > 0)) { if (!NameIsDefined(tmp, o, out realName)) throw new MissingMemberException(o.GetType().Name, tmp); // // Attempt to match default, no parameter (if overloaded) // Indexer accesses will fail - we do not handle indexers // Do case sensitive here because we have the real name of the matching member val = t.InvokeMember(realName, BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.GetProperty | BindingFlags.GetField | BindingFlags.Instance | BindingFlags.Static, null, o, null, System.Globalization.CultureInfo.InvariantCulture); } else val = o; // root object is a collection - all that is passed for name is "[1]" if (isCollection) { IEnumerable collection = val as IEnumerable; if (null != collection) GetEnumerationMember(collection, index, out obj); } else obj = val; } internal static void GetEnumerationMember(IEnumerable collection, int index, out object obj) { obj = null; if (null == collection) throw new ArgumentNullException("collection"); IEnumerator e = collection.GetEnumerator(); int i = 0; while (e.MoveNext()) { if (i++ == index) { obj = e.Current; return; } } throw new IndexOutOfRangeException(); } internal static object GetRuntimeValue(object o, Activity activity) { if (null == o) return o; object tmp = o; if (o is ActivityBind) { if (null == activity) throw new ArgumentNullException("activity"); tmp = ((ActivityBind)o).GetRuntimeValue(activity); } else if (o is WorkflowParameterBinding) { tmp = ((WorkflowParameterBinding)o).Value; } return tmp; } internal static void GetAllMembers(Activity activity, IList items, TrackingAnnotationCollection annotations) { Type t = activity.GetType(); // // Get all fields FieldInfo[] fields = t.GetFields(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.Static | BindingFlags.GetField); foreach (FieldInfo f in fields) { if (!PropertyHelper.IsInternalVariable(f.Name)) { TrackingDataItem data = new TrackingDataItem(); data.FieldName = f.Name; data.Data = GetRuntimeValue(f.GetValue(activity), activity); foreach (string s in annotations) data.Annotations.Add(s); items.Add(data); } } // // Get all properties (except indexers) PropertyInfo[] properties = t.GetProperties(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.Static | BindingFlags.GetProperty); foreach (PropertyInfo p in properties) { if (!IsInternalVariable(p.Name)) { // // Skip indexers, since private data members // are exposed the data is still available. if (p.GetIndexParameters().Length > 0) continue; TrackingDataItem data = new TrackingDataItem(); data.FieldName = p.Name; data.Data = GetRuntimeValue(p.GetValue(activity, null), activity); foreach (string s in annotations) data.Annotations.Add(s); items.Add(data); } } } #endregion #region Private Methods private static bool IsInternalVariable(string name) { string[] vars = { "__winoe_ActivityLocks_", "__winoe_StaticActivityLocks_", "__winoe_MethodLocks_" }; foreach (string s in vars) { if (0 == string.Compare(s, name, StringComparison.Ordinal)) return true; } return false; } private static bool NameIsDefined(string name, object o, out string realName) { realName = null; Type t = o.GetType(); // // Get the member with the requested name // Do case specific first MemberInfo[] members = t.GetMember(name, BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.GetProperty | BindingFlags.GetField | BindingFlags.Instance | BindingFlags.Static); // // Not found if ((null == members) || (0 == members.Length)) { // // Do case insensitive members = t.GetMember(name, BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.GetProperty | BindingFlags.GetField | BindingFlags.Instance | BindingFlags.Static | BindingFlags.IgnoreCase); // // Not found if ((null == members) || (0 == members.Length)) { return false; } } if ((null == members) || (0 == members.Length) || (null == members[0].Name) || (0 == members[0].Name.Length)) return false; realName = members[0].Name; return true; } private static bool TryParseIndex(string fullName, out string name, out int index) { name = null; index = -1; int endPos = -1, startPos = -1; for (int i = fullName.Length - 1; i > 0; i--) { if ((']' == fullName[i]) && (-1 == endPos)) { endPos = i; } else if (('[' == fullName[i]) && (-1 == startPos)) { startPos = i; break; } } if ((-1 == endPos) || (-1 == startPos)) return false; string idx = fullName.Substring(startPos + 1, endPos - 1 - startPos); name = fullName.Substring(0, startPos); return int.TryParse(idx, out index); } #endregion } }