mirror of https://github.com/OpenIdentityPlatform/OpenDJ.git

matthew_swift
28.47.2010 f2160f4bd1c8ac67e5a86a6710d431e8932877f9
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License, Version 1.0 only
 * (the "License").  You may not use this file except in compliance
 * with the License.
 *
 * You can obtain a copy of the license at
 * trunk/opends/resource/legal-notices/OpenDS.LICENSE
 * or https://OpenDS.dev.java.net/OpenDS.LICENSE.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at
 * trunk/opends/resource/legal-notices/OpenDS.LICENSE.  If applicable,
 * add the following below this CDDL HEADER, with the fields enclosed
 * by brackets "[]" replaced with your own identifying information:
 *      Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 *
 *
 *      Copyright 2009 Sun Microsystems, Inc.
 */
 
package org.opends.sdk;
 
 
 
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.util.IdentityHashMap;
import java.util.Set;
 
import org.testng.annotations.AfterClass;
import org.testng.annotations.Test;
 
 
 
/**
 * This class defines a base test case that should be subclassed by all unit
 * tests used by OpenDS.
 * <p>
 * This class adds the ability to print error messages and automatically have
 * them include the class name.
 */
@Test(sequential = true)
public abstract class OpenDSTestCase
{
 
  //
  // This is all a HACK to reduce the amount of memory that's consumed.
  //
  // This could be a problem if a subclass references a @DataProvider in
  // a super-class that provides static parameters, i.e. the parameters
  // are
  // not regenerated for each invocation of the DataProvider.
  //
 
  /**
   * A list of all parameters that were generated by a @DataProvider and passed
   * to a test method of this class. TestListener helps us keep this so that
   * once all of the tests are finished, we can clear it out in an @AfterClass
   * method. We can't just clear it out right away in the TestListener because
   * some methods share a @DataProvider.
   */
  private final IdentityHashMap<Object[], Object> successfulTestParams = new IdentityHashMap<Object[], Object>();
 
  /**
   * These are test parameters from a test that has failed. We need to keep
   * these around because the test report expects to find them when printing out
   * failures.
   */
  private final IdentityHashMap<Object[], Object> failedTestParams = new IdentityHashMap<Object[], Object>();
 
 
 
  /**
   * null out all test parameters except the ones used in failed tests since we
   * might need these again.
   */
  @AfterClass(alwaysRun = true)
  public void clearSuccessfulTestParams()
  {
    final Set<Object[]> paramsSet = successfulTestParams.keySet();
    if (paramsSet == null)
    { // Can this ever happen?
      return;
    }
    for (final Object[] params : paramsSet)
    {
      if (failedTestParams.containsKey(params))
      {
        continue;
      }
 
      for (int i = 0; i < params.length; i++)
      {
        params[i] = null;
      }
    }
    successfulTestParams.clear();
    failedTestParams.clear();
  }
 
 
 
  /**
   * The member variables of a test class can prevent lots of memory from being
   * reclaimed, so we use reflection to null out all of the member variables
   * after the tests have run. Since all tests must inherit from
   * DirectoryServerTestCase, TestNG guarantees that this method runs after all
   * of the subclass methods, so this isn't too dangerous.
   */
  @AfterClass(alwaysRun = true)
  public void nullMemberVariablesAfterTest()
  {
    Class<?> cls = this.getClass();
    // Iterate through all of the fields in all subclasses of
    // DirectoryServerTestCase, but not DirectoryServerTestCase itself.
    while (OpenDSTestCase.class.isAssignableFrom(cls)
        && !OpenDSTestCase.class.equals(cls))
    {
      final Field fields[] = cls.getDeclaredFields();
      for (final Field field : fields)
      {
        final int modifiers = field.getModifiers();
        final Class<?> fieldClass = field.getType();
        // If it's a non-static non-final non-primitive type, then null
        // it out
        // so that the garbage collector can reclaim it and everything
        // it
        // references.
        if (!fieldClass.isPrimitive() && !fieldClass.isEnum()
            && !Modifier.isFinal(modifiers) && !Modifier.isStatic(modifiers))
        {
          field.setAccessible(true);
          try
          {
            field.set(this, null);
          }
          catch (final IllegalAccessException e)
          {
            // We're only doing this to save memory, so it's no big deal
            // if we can't set it.
          }
        }
      }
      cls = cls.getSuperclass();
    }
  }
 
 
 
  /**
   * Adds testParams to the list of all failed test parameters, so that we know
   * to NOT null it out later.
   */
  void addParamsFromFailedTest(final Object[] testParams)
  {
    if (testParams != null)
    {
      failedTestParams.put(testParams, testParams);
    }
  }
 
 
 
  /**
   * Adds testParams to the list of all test parameters, so it can be null'ed
   * out later if it's not part.
   */
  void addParamsFromSuccessfulTests(final Object[] testParams)
  {
    if (testParams != null)
    {
      successfulTestParams.put(testParams, testParams);
    }
  }
}