/IronPython_2_6/Src/Runtime/Microsoft.Dynamic/Actions/DefaultBinder.DeleteMember.cs

# · C# · 140 lines · 99 code · 23 blank · 18 comment · 19 complexity · b8f636c16a0fd933b58f0113727867e2 MD5 · raw file

  1. /* ****************************************************************************
  2. *
  3. * Copyright (c) Microsoft Corporation.
  4. *
  5. * This source code is subject to terms and conditions of the Microsoft Public License. A
  6. * copy of the license can be found in the License.html file at the root of this distribution. If
  7. * you cannot locate the Microsoft Public License, please send an email to
  8. * dlr@microsoft.com. By using this source code in any fashion, you are agreeing to be bound
  9. * by the terms of the Microsoft Public License.
  10. *
  11. * You must not remove this notice, or any other, from this software.
  12. *
  13. *
  14. * ***************************************************************************/
  15. using System;
  16. using System.Dynamic;
  17. using System.Reflection;
  18. using Microsoft.Scripting.Actions.Calls;
  19. using Microsoft.Scripting.Runtime;
  20. using Microsoft.Scripting.Utils;
  21. using AstUtils = Microsoft.Scripting.Ast.Utils;
  22. namespace Microsoft.Scripting.Actions {
  23. public partial class DefaultBinder : ActionBinder {
  24. public DynamicMetaObject DeleteMember(string name, DynamicMetaObject target) {
  25. return DeleteMember(name, target, new DefaultOverloadResolverFactory(this));
  26. }
  27. public DynamicMetaObject DeleteMember(string name, DynamicMetaObject target, OverloadResolverFactory resolutionFactory) {
  28. return DeleteMember(name, target, resolutionFactory, null);
  29. }
  30. public DynamicMetaObject DeleteMember(string name, DynamicMetaObject target, OverloadResolverFactory resolutionFactory, DynamicMetaObject errorSuggestion) {
  31. ContractUtils.RequiresNotNull(name, "name");
  32. ContractUtils.RequiresNotNull(target, "target");
  33. return MakeDeleteMemberTarget(
  34. new SetOrDeleteMemberInfo(
  35. name,
  36. resolutionFactory
  37. ),
  38. target.Restrict(target.GetLimitType()),
  39. errorSuggestion
  40. );
  41. }
  42. private DynamicMetaObject MakeDeleteMemberTarget(SetOrDeleteMemberInfo delInfo, DynamicMetaObject target, DynamicMetaObject errorSuggestion) {
  43. Type type = target.GetLimitType();
  44. BindingRestrictions restrictions = target.Restrictions;
  45. DynamicMetaObject self = target;
  46. if (typeof(TypeTracker).IsAssignableFrom(type)) {
  47. restrictions = restrictions.Merge(
  48. BindingRestrictions.GetInstanceRestriction(target.Expression, target.Value)
  49. );
  50. type = ((TypeTracker)target.Value).Type;
  51. self = null;
  52. }
  53. delInfo.Body.Restrictions = restrictions;
  54. if (self == null || !MakeOperatorDeleteMemberBody(delInfo, self, type, "DeleteMember")) {
  55. MemberGroup group = GetMember(MemberRequestKind.Delete, type, delInfo.Name);
  56. if (group.Count != 0) {
  57. if (group[0].MemberType == TrackerTypes.Property) {
  58. MethodInfo del = ((PropertyTracker)group[0]).GetDeleteMethod(PrivateBinding);
  59. if (del != null) {
  60. MakePropertyDeleteStatement(delInfo, self, del);
  61. return delInfo.Body.GetMetaObject(target);
  62. }
  63. }
  64. delInfo.Body.FinishCondition(errorSuggestion ?? MakeError(MakeUndeletableMemberError(GetDeclaringMemberType(group), delInfo.Name), typeof(void)));
  65. } else {
  66. delInfo.Body.FinishCondition(errorSuggestion ?? MakeError(MakeMissingMemberErrorForDelete(type, self, delInfo.Name), typeof(void)));
  67. }
  68. }
  69. return delInfo.Body.GetMetaObject(target);
  70. }
  71. private static Type GetDeclaringMemberType(MemberGroup group) {
  72. Type t = typeof(object);
  73. foreach (MemberTracker mt in group) {
  74. if (t.IsAssignableFrom(mt.DeclaringType)) {
  75. t = mt.DeclaringType;
  76. }
  77. }
  78. return t;
  79. }
  80. private void MakePropertyDeleteStatement(SetOrDeleteMemberInfo delInfo, DynamicMetaObject instance, MethodInfo delete) {
  81. delInfo.Body.FinishCondition(
  82. instance == null ?
  83. MakeCallExpression(delInfo.ResolutionFactory, delete) :
  84. MakeCallExpression(delInfo.ResolutionFactory, delete, instance)
  85. );
  86. }
  87. /// <summary> if a member-injector is defined-on or registered-for this type call it </summary>
  88. private bool MakeOperatorDeleteMemberBody(SetOrDeleteMemberInfo delInfo, DynamicMetaObject instance, Type type, string name) {
  89. MethodInfo delMem = GetMethod(type, name);
  90. if (delMem != null) {
  91. DynamicMetaObject call = MakeCallExpression(delInfo.ResolutionFactory, delMem, instance, new DynamicMetaObject(AstUtils.Constant(delInfo.Name), BindingRestrictions.Empty, delInfo.Name));
  92. if (delMem.ReturnType == typeof(bool)) {
  93. delInfo.Body.AddCondition(
  94. call.Expression,
  95. AstUtils.Constant(null)
  96. );
  97. } else {
  98. delInfo.Body.FinishCondition(call);
  99. }
  100. return delMem.ReturnType != typeof(bool);
  101. }
  102. return false;
  103. }
  104. /// <summary>
  105. /// Helper class for flowing information about the GetMember request.
  106. /// </summary>
  107. private sealed class SetOrDeleteMemberInfo {
  108. public readonly string Name;
  109. public readonly OverloadResolverFactory ResolutionFactory;
  110. public readonly ConditionalBuilder Body = new ConditionalBuilder();
  111. public SetOrDeleteMemberInfo(string name, OverloadResolverFactory resolutionFactory) {
  112. Name = name;
  113. ResolutionFactory = resolutionFactory;
  114. }
  115. }
  116. }
  117. }