Ensure that RandomStringUtils returns all expected characters

bug 20592, reported/patched by Phil Steitz


git-svn-id: https://svn.apache.org/repos/asf/jakarta/commons/proper/lang/trunk@137365 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Stephen Colebourne 2003-06-09 21:36:03 +00:00
parent 8955311526
commit 8f45918465
3 changed files with 123 additions and 30 deletions

View File

@ -63,8 +63,9 @@ import java.util.Random;
* @author <a href="mailto:steven@caswell.name">Steven Caswell</a> * @author <a href="mailto:steven@caswell.name">Steven Caswell</a>
* @author Stephen Colebourne * @author Stephen Colebourne
* @author <a href="mailto:ggregory@seagullsw.com">Gary Gregory</a> * @author <a href="mailto:ggregory@seagullsw.com">Gary Gregory</a>
* @author Phil Steitz
* @since 1.0 * @since 1.0
* @version $Id: RandomStringUtils.java,v 1.12 2003/05/20 21:15:19 ggregory Exp $ * @version $Id: RandomStringUtils.java,v 1.13 2003/06/09 21:36:02 scolebourne Exp $
*/ */
public class RandomStringUtils { public class RandomStringUtils {
@ -104,7 +105,7 @@ public class RandomStringUtils {
* specified.</p> * specified.</p>
* *
* <p>Characters will be chosen from the set of characters whose * <p>Characters will be chosen from the set of characters whose
* ASCII value is between <code>32</code> and <code>127</code>.</p> * ASCII value is between <code>32</code> and <code>126</code> (inclusive).</p>
* *
* @param count length of random string to create * @param count length of random string to create
* @return the random string * @return the random string
@ -199,9 +200,8 @@ public class RandomStringUtils {
* *
* This method has exactly the same semantics as {@link * This method has exactly the same semantics as {@link
* #random(int,int,int,boolean,boolean,char[],Random)}, but * #random(int,int,int,boolean,boolean,char[],Random)}, but
* instead of depending on internal source of randomness ({@link * instead of using an externally supplied source of randomness, it uses
* #RANDOM}) it uses externally supplied instance of {@link * the internal static {@link Random} instance ({@link #RANDOM}).
* Random} class.
* *
* @param count length of random string to create * @param count length of random string to create
* @param start position in set of chars to start at * @param start position in set of chars to start at
@ -256,7 +256,7 @@ public class RandomStringUtils {
throw new IllegalArgumentException("Requested random string length " + count + " is less than 0."); throw new IllegalArgumentException("Requested random string length " + count + " is less than 0.");
} }
if( (start == 0) && (end == 0) ) { if( (start == 0) && (end == 0) ) {
end = (int)'z'; end = (int)'z' + 1;
start = (int)' '; start = (int)' ';
if(!letters && !numbers) { if(!letters && !numbers) {
start = 0; start = 0;

View File

@ -57,12 +57,14 @@ import java.util.Random;
import junit.framework.*; import junit.framework.*;
import junit.textui.TestRunner; import junit.textui.TestRunner;
/** /**
* Unit tests {@link org.apache.commons.lang.RandomStringUtils}. * Unit tests {@link org.apache.commons.lang.RandomStringUtils}.
* *
* @author <a href="mailto:steven@caswell.name">Steven Caswell</a> * @author <a href="mailto:steven@caswell.name">Steven Caswell</a>
* @author <a href="mailto:ridesmet@users.sourceforge.net">Ringo De Smet</a> * @author <a href="mailto:ridesmet@users.sourceforge.net">Ringo De Smet</a>
* @version $Id: RandomStringUtilsTest.java,v 1.6 2003/05/14 02:50:43 bayard Exp $ * @author Phil Steitz
* @version $Id: RandomStringUtilsTest.java,v 1.7 2003/06/09 21:36:03 scolebourne Exp $
*/ */
public class RandomStringUtilsTest extends junit.framework.TestCase { public class RandomStringUtilsTest extends junit.framework.TestCase {
/** /**
@ -167,6 +169,99 @@ public class RandomStringUtilsTest extends junit.framework.TestCase {
assertNotNull("random(<0) throws exception", e); assertNotNull("random(<0) throws exception", e);
} }
/**
* Make sure boundary alphanumeric characters are generated by randomAlphaNumeric
* This test will fail randomly with probability = 6 * (61/62)**1000 ~ 5.2E-7
*/
public void testRandomAlphaNumeric() {
char[] testChars = {'a', 'z', 'A', 'Z', '0', '9'};
boolean[] found = {false, false, false, false, false, false};
for (int i = 0; i < 100; i++) {
String randString = RandomStringUtils.randomAlphanumeric(10);
for (int j = 0; j < testChars.length; j++) {
if (randString.indexOf(testChars[j]) > 0) {
found[j] = true;
}
}
}
for (int i = 0; i < testChars.length; i++) {
if (!found[i]) {
fail("alphanumeric character not generated in 1000 attempts: "
+ testChars[i] +" -- repeated failures indicate a problem ");
}
}
}
/**
* Make sure '0' and '9' are generated by randomNumeric
* This test will fail randomly with probability = 2 * (9/10)**1000 ~ 3.5E-46
*/
public void testRandomNumeric() {
char[] testChars = {'0','9'};
boolean[] found = {false, false};
for (int i = 0; i < 100; i++) {
String randString = RandomStringUtils.randomNumeric(10);
for (int j = 0; j < testChars.length; j++) {
if (randString.indexOf(testChars[j]) > 0) {
found[j] = true;
}
}
}
for (int i = 0; i < testChars.length; i++) {
if (!found[i]) {
fail("digit not generated in 1000 attempts: "
+ testChars[i] +" -- repeated failures indicate a problem ");
}
}
}
/**
* Make sure boundary alpha characters are generated by randomAlphabetic
* This test will fail randomly with probability = 4 * (51/52)**1000 ~ 1.58E-8
*/
public void testRandomAlphabetic() {
char[] testChars = {'a', 'z', 'A', 'Z'};
boolean[] found = {false, false, false, false};
for (int i = 0; i < 100; i++) {
String randString = RandomStringUtils.randomAlphabetic(10);
for (int j = 0; j < testChars.length; j++) {
if (randString.indexOf(testChars[j]) > 0) {
found[j] = true;
}
}
}
for (int i = 0; i < testChars.length; i++) {
if (!found[i]) {
fail("alphanumeric character not generated in 1000 attempts: "
+ testChars[i] +" -- repeated failures indicate a problem ");
}
}
}
/**
* Make sure 32 and 127 are generated by randomNumeric
* This test will fail randomly with probability = 2*(95/96)**1000 ~ 5.7E-5
*/
public void testRandomAscii() {
char[] testChars = {(char) 32, (char) 126};
boolean[] found = {false, false};
for (int i = 0; i < 100; i++) {
String randString = RandomStringUtils.randomAscii(10);
for (int j = 0; j < testChars.length; j++) {
if (randString.indexOf(testChars[j]) > 0) {
found[j] = true;
}
}
}
for (int i = 0; i < testChars.length; i++) {
if (!found[i]) {
fail("ascii character not generated in 1000 attempts: "
+ (int) testChars[i] +
" -- repeated failures indicate a problem");
}
}
}
/** /**
* Test homogeneity of random strings generated -- * Test homogeneity of random strings generated --
* i.e., test that characters show up with expected frequencies * i.e., test that characters show up with expected frequencies
@ -195,8 +290,6 @@ public class RandomStringUtilsTest extends junit.framework.TestCase {
chiSquare(expected,counts) < 13.82); chiSquare(expected,counts) < 13.82);
} }
//FIXME: add similar tests for other functions
/** /**
* Computes Chi-Square statistic given observed and expected counts * Computes Chi-Square statistic given observed and expected counts
* @param observed array of observed frequency counts * @param observed array of observed frequency counts

View File

@ -63,7 +63,7 @@ import junit.framework.TestSuite;
* Test cases for the {@link RandomUtils} class. * Test cases for the {@link RandomUtils} class.
* *
* @author <a href="mailto:phil@steitz.com">Phil Steitz</a> * @author <a href="mailto:phil@steitz.com">Phil Steitz</a>
* @version $Revision: 1.2 $ $Date: 2003/06/08 14:19:43 $ * @version $Revision: 1.3 $ $Date: 2003/06/09 21:36:03 $
*/ */
public final class RandomUtilsTest extends TestCase { public final class RandomUtilsTest extends TestCase {
@ -126,12 +126,12 @@ public final class RandomUtilsTest extends TestCase {
assertTrue(result >= 0); assertTrue(result >= 0);
observed[result]++; observed[result]++;
} }
/* Use ChiSquare dist with df = 4-1 = 3, alpha = .01 /* Use ChiSquare dist with df = 4-1 = 3, alpha = .001
* Change to 16.27 for alpha = .001 * Change to 11.34 for alpha = .01
*/ */
assertTrue( assertTrue(
"chi-square test -- will fail about 1 in 100 times", "chi-square test -- will fail about 1 in 1000 times",
chiSquare(expected,observed) < 11.34); chiSquare(expected,observed) < 16.27);
} }
/** test distribution of nextLong() */ /** test distribution of nextLong() */
@ -171,12 +171,12 @@ public final class RandomUtilsTest extends TestCase {
observed[1]++; observed[1]++;
} }
} }
/* Use ChiSquare dist with df = 2-1 = 1, alpha = .01 /* Use ChiSquare dist with df = 2-1 = 1, alpha = .001
* Change to 10.83 for alpha = .001 * Change to 6.64 for alpha = .01
*/ */
assertTrue( assertTrue(
"chi-square test -- will fail about 1 in 100 times", "chi-square test -- will fail about 1 in 1000 times",
chiSquare(expected,observed) < 6.64); chiSquare(expected,observed) < 10.83);
} }
@ -214,12 +214,12 @@ public final class RandomUtilsTest extends TestCase {
observed[1]++; observed[1]++;
} }
} }
/* Use ChiSquare dist with df = 2-1 = 1, alpha = .01 /* Use ChiSquare dist with df = 2-1 = 1, alpha = .001
* Change to 10.83 for alpha = .001 * Change to 6.64 for alpha = .01
*/ */
assertTrue( assertTrue(
"chi-square test -- will fail about 1 in 100 times", "chi-square test -- will fail about 1 in 1000 times",
chiSquare(expected,observed) < 6.64); chiSquare(expected,observed) < 10.83 );
} }
/** test distribution of nextFloat() */ /** test distribution of nextFloat() */
@ -256,12 +256,12 @@ public final class RandomUtilsTest extends TestCase {
observed[1]++; observed[1]++;
} }
} }
/* Use ChiSquare dist with df = 2-1 = 1, alpha = .01 /* Use ChiSquare dist with df = 2-1 = 1, alpha = .001
* Change to 10.83 for alpha = .001 * Change to 6.64 for alpha = .01
*/ */
assertTrue( assertTrue(
"chi-square test -- will fail about 1 in 100 times", "chi-square test -- will fail about 1 in 1000 times",
chiSquare(expected,observed) < 6.64); chiSquare(expected,observed) < 10.83);
} }
/** test distribution of nextDouble() */ /** test distribution of nextDouble() */
@ -298,12 +298,12 @@ public final class RandomUtilsTest extends TestCase {
observed[1]++; observed[1]++;
} }
} }
/* Use ChiSquare dist with df = 2-1 = 1, alpha = .01 /* Use ChiSquare dist with df = 2-1 = 1, alpha = .001
* Change to 10.83 for alpha = .001 * Change to 6.64 for alpha = .01
*/ */
assertTrue( assertTrue(
"chi-square test -- will fail about 1 in 100 times", "chi-square test -- will fail about 1 in 1000 times",
chiSquare(expected,observed) < 6.64); chiSquare(expected,observed) < 10.83);
} }
/** make sure that setSeed fails */ /** make sure that setSeed fails */