RandomStringUtils.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.lang3;

  18. import java.security.SecureRandom;
  19. import java.security.Security;
  20. import java.util.Random;
  21. import java.util.concurrent.ThreadLocalRandom;
  22. import java.util.function.Supplier;

  23. /**
  24.  * Generates random {@link String}s.
  25.  * <p>
  26.  * Use {@link #secure()} to get the singleton instance based on {@link SecureRandom#SecureRandom()} which uses a secure random number generator implementing the
  27.  * default random number algorithm.
  28.  * </p>
  29.  * <p>
  30.  * Use {@link #secureStrong()} to get the singleton instance based on {@link SecureRandom#getInstanceStrong()} which uses an instance that was selected by using
  31.  * the algorithms/providers specified in the {@code securerandom.strongAlgorithms} {@link Security} property.
  32.  * </p>
  33.  * <p>
  34.  * Use {@link #insecure()} to get the singleton instance based on {@link ThreadLocalRandom#current()} <b>which is not cryptographically secure</b>. In addition,
  35.  * instances do not use a cryptographically random seed unless the {@linkplain System#getProperty system property} {@code java.util.secureRandomSeed} is set to
  36.  * {@code true}.
  37.  * </p>
  38.  * <p>
  39.  * Starting in version 3.17.0, the method {@link #secure()} uses {@link SecureRandom#SecureRandom()} instead of {@link SecureRandom#getInstanceStrong()}, and
  40.  * adds {@link #secureStrong()}.
  41.  * </p>
  42.  * <p>
  43.  * Starting in version 3.16.0, this class uses {@link #secure()} for static methods and adds {@link #insecure()}.
  44.  * </p>
  45.  * <p>
  46.  * Starting in version 3.15.0, this class uses {@link SecureRandom#getInstanceStrong()} for static methods.
  47.  * </p>
  48.  * <p>
  49.  * Before version 3.15.0, this class used {@link ThreadLocalRandom#current()} for static methods, which is not cryptographically secure.
  50.  * </p>
  51.  * <p>
  52.  * RandomStringUtils is intended for simple use cases. For more advanced use cases consider using Apache Commons Text's
  53.  * <a href= "https://commons.apache.org/proper/commons-text/javadocs/api-release/org/apache/commons/text/RandomStringGenerator.html"> RandomStringGenerator</a>
  54.  * instead.
  55.  * </p>
  56.  * <p>
  57.  * The Apache Commons project provides <a href="https://commons.apache.org/proper/commons-rng/">Commons RNG</a> dedicated to pseudo-random number generation,
  58.  * that may be a better choice for applications with more stringent requirements (performance and/or correctness).
  59.  * </p>
  60.  * <p>
  61.  * Note that <em>private high surrogate</em> characters are ignored. These are Unicode characters that fall between the values 56192 (db80) and 56319 (dbff) as
  62.  * we don't know how to handle them. High and low surrogates are correctly dealt with - that is if a high surrogate is randomly chosen, 55296 (d800) to 56191
  63.  * (db7f) then it is followed by a low surrogate. If a low surrogate is chosen, 56320 (dc00) to 57343 (dfff) then it is placed after a randomly chosen high
  64.  * surrogate.
  65.  * </p>
  66.  * <p>
  67.  * #ThreadSafe#
  68.  * </p>
  69.  *
  70.  * @see #secure()
  71.  * @see #secureStrong()
  72.  * @see #insecure()
  73.  * @see SecureRandom#SecureRandom()
  74.  * @see SecureRandom#getInstanceStrong()
  75.  * @see ThreadLocalRandom#current()
  76.  * @see RandomUtils
  77.  * @since 1.0
  78.  */
  79. public class RandomStringUtils {

  80.     private static final Supplier<RandomUtils> SECURE_SUPPLIER = RandomUtils::secure;

  81.     private static RandomStringUtils INSECURE = new RandomStringUtils(RandomUtils::insecure);

  82.     private static RandomStringUtils SECURE = new RandomStringUtils(SECURE_SUPPLIER);

  83.     private static RandomStringUtils SECURE_STRONG = new RandomStringUtils(RandomUtils::secureStrong);

  84.     private static final char[] ALPHANUMERICAL_CHARS = { 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l',
  85.             'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', 'A', 'B', 'C', 'D', 'E', 'F', 'G',
  86.             'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', '0', '1',
  87.             '2', '3', '4', '5', '6', '7', '8', '9' };

  88.     /**
  89.      * Gets the singleton instance based on {@link ThreadLocalRandom#current()}; <b>which is not cryptographically
  90.      * secure</b>; use {@link #secure()} to use an algorithms/providers specified in the
  91.      * {@code securerandom.strongAlgorithms} {@link Security} property.
  92.      * <p>
  93.      * The method {@link ThreadLocalRandom#current()} is called on-demand.
  94.      * </p>
  95.      *
  96.      * @return the singleton instance based on {@link ThreadLocalRandom#current()}.
  97.      * @see ThreadLocalRandom#current()
  98.      * @see #secure()
  99.      * @since 3.16.0
  100.      */
  101.     public static RandomStringUtils insecure() {
  102.         return INSECURE;
  103.     }

  104.     /**
  105.      * Creates a random string whose length is the number of characters specified.
  106.      *
  107.      * <p>
  108.      * Characters will be chosen from the set of all characters.
  109.      * </p>
  110.      *
  111.      * @param count the length of random string to create
  112.      * @return the random string
  113.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  114.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  115.      */
  116.     @Deprecated
  117.     public static String random(final int count) {
  118.         return secure().next(count);
  119.     }

  120.     /**
  121.      * Creates a random string whose length is the number of characters specified.
  122.      *
  123.      * <p>
  124.      * Characters will be chosen from the set of alpha-numeric characters as indicated by the arguments.
  125.      * </p>
  126.      *
  127.      * @param count   the length of random string to create
  128.      * @param letters if {@code true}, generated string may include alphabetic characters
  129.      * @param numbers if {@code true}, generated string may include numeric characters
  130.      * @return the random string
  131.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  132.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  133.      */
  134.     @Deprecated
  135.     public static String random(final int count, final boolean letters, final boolean numbers) {
  136.         return secure().next(count, letters, numbers);
  137.     }

  138.     /**
  139.      * Creates a random string whose length is the number of characters specified.
  140.      *
  141.      * <p>
  142.      * Characters will be chosen from the set of characters specified.
  143.      * </p>
  144.      *
  145.      * @param count the length of random string to create
  146.      * @param chars the character array containing the set of characters to use, may be null
  147.      * @return the random string
  148.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  149.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  150.      */
  151.     @Deprecated
  152.     public static String random(final int count, final char... chars) {
  153.         return secure().next(count, chars);
  154.     }

  155.     /**
  156.      * Creates a random string whose length is the number of characters specified.
  157.      *
  158.      * <p>
  159.      * Characters will be chosen from the set of alpha-numeric characters as indicated by the arguments.
  160.      * </p>
  161.      *
  162.      * @param count   the length of random string to create
  163.      * @param start   the position in set of chars to start at
  164.      * @param end     the position in set of chars to end before
  165.      * @param letters if {@code true}, generated string may include alphabetic characters
  166.      * @param numbers if {@code true}, generated string may include numeric characters
  167.      * @return the random string
  168.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  169.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  170.      */
  171.     @Deprecated
  172.     public static String random(final int count, final int start, final int end, final boolean letters,
  173.             final boolean numbers) {
  174.         return secure().next(count, start, end, letters, numbers);
  175.     }

  176.     /**
  177.      * Creates a random string based on a variety of options, using default source of randomness.
  178.      *
  179.      * <p>
  180.      * This method has exactly the same semantics as {@link #random(int,int,int,boolean,boolean,char[],Random)}, but
  181.      * instead of using an externally supplied source of randomness, it uses the internal static {@link Random}
  182.      * instance.
  183.      * </p>
  184.      *
  185.      * @param count   the length of random string to create
  186.      * @param start   the position in set of chars to start at
  187.      * @param end     the position in set of chars to end before
  188.      * @param letters if {@code true}, generated string may include alphabetic characters
  189.      * @param numbers if {@code true}, generated string may include numeric characters
  190.      * @param chars   the set of chars to choose randoms from. If {@code null}, then it will use the set of all chars.
  191.      * @return the random string
  192.      * @throws ArrayIndexOutOfBoundsException if there are not {@code (end - start) + 1} characters in the set array.
  193.      * @throws IllegalArgumentException       if {@code count} &lt; 0.
  194.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  195.      */
  196.     @Deprecated
  197.     public static String random(final int count, final int start, final int end, final boolean letters,
  198.             final boolean numbers, final char... chars) {
  199.         return secure().next(count, start, end, letters, numbers, chars);
  200.     }

  201.     /**
  202.      * Creates a random string based on a variety of options, using supplied source of randomness.
  203.      *
  204.      * <p>
  205.      * If start and end are both {@code 0}, start and end are set to {@code ' '} and {@code 'z'}, the ASCII printable
  206.      * characters, will be used, unless letters and numbers are both {@code false}, in which case, start and end are set
  207.      * to {@code 0} and {@link Character#MAX_CODE_POINT}.
  208.      *
  209.      * <p>
  210.      * If set is not {@code null}, characters between start and end are chosen.
  211.      * </p>
  212.      *
  213.      * <p>
  214.      * This method accepts a user-supplied {@link Random} instance to use as a source of randomness. By seeding a single
  215.      * {@link Random} instance with a fixed seed and using it for each call, the same random sequence of strings can be
  216.      * generated repeatedly and predictably.
  217.      * </p>
  218.      *
  219.      * @param count   the length of random string to create
  220.      * @param start   the position in set of chars to start at (inclusive)
  221.      * @param end     the position in set of chars to end before (exclusive)
  222.      * @param letters if {@code true}, generated string may include alphabetic characters
  223.      * @param numbers if {@code true}, generated string may include numeric characters
  224.      * @param chars   the set of chars to choose randoms from, must not be empty. If {@code null}, then it will use the
  225.      *                set of all chars.
  226.      * @param random  a source of randomness.
  227.      * @return the random string
  228.      * @throws ArrayIndexOutOfBoundsException if there are not {@code (end - start) + 1} characters in the set array.
  229.      * @throws IllegalArgumentException       if {@code count} &lt; 0 or the provided chars array is empty.
  230.      * @since 2.0
  231.      */
  232.     public static String random(int count, int start, int end, final boolean letters, final boolean numbers,
  233.             final char[] chars, final Random random) {
  234.         if (count == 0) {
  235.             return StringUtils.EMPTY;
  236.         }
  237.         if (count < 0) {
  238.             throw new IllegalArgumentException("Requested random string length " + count + " is less than 0.");
  239.         }
  240.         if (chars != null && chars.length == 0) {
  241.             throw new IllegalArgumentException("The chars array must not be empty");
  242.         }

  243.         if (start == 0 && end == 0) {
  244.             if (chars != null) {
  245.                 end = chars.length;
  246.             } else if (!letters && !numbers) {
  247.                 end = Character.MAX_CODE_POINT;
  248.             } else {
  249.                 end = 'z' + 1;
  250.                 start = ' ';
  251.             }
  252.         } else if (end <= start) {
  253.             throw new IllegalArgumentException(
  254.                     "Parameter end (" + end + ") must be greater than start (" + start + ")");
  255.         } else if (start < 0 || end < 0) {
  256.             throw new IllegalArgumentException("Character positions MUST be >= 0");
  257.         }

  258.         if (end > Character.MAX_CODE_POINT) {
  259.             // Technically, it should be `Character.MAX_CODE_POINT+1` as `end` is excluded
  260.             // But the character `Character.MAX_CODE_POINT` is private use, so it would anyway be excluded
  261.             end = Character.MAX_CODE_POINT;
  262.         }

  263.         // Optimize generation of full alphanumerical characters
  264.         // Normally, we would need to pick a 7-bit integer, since gap = 'z' - '0' + 1 = 75 > 64
  265.         // In turn, this would make us reject the sampling with probability 1 - 62 / 2^7 > 1 / 2
  266.         // Instead we can pick directly from the right set of 62 characters, which requires
  267.         // picking a 6-bit integer and only rejecting with probability 2 / 64 = 1 / 32
  268.         if (chars == null && letters && numbers && start <= '0' && end >= 'z' + 1) {
  269.             return random(count, 0, 0, false, false, ALPHANUMERICAL_CHARS, random);
  270.         }

  271.         // Optimize start and end when filtering by letters and/or numbers:
  272.         // The range provided may be too large since we filter anyway afterward.
  273.         // Note the use of Math.min/max (as opposed to setting start to '0' for example),
  274.         // since it is possible the range start/end excludes some of the letters/numbers,
  275.         // e.g., it is possible that start already is '1' when numbers = true, and start
  276.         // needs to stay equal to '1' in that case.
  277.         if (chars == null) {
  278.             if (letters && numbers) {
  279.                 start = Math.max('0', start);
  280.                 end = Math.min('z' + 1, end);
  281.             } else if (numbers) {
  282.                 // just numbers, no letters
  283.                 start = Math.max('0', start);
  284.                 end = Math.min('9' + 1, end);
  285.             } else if (letters) {
  286.                 // just letters, no numbers
  287.                 start = Math.max('A', start);
  288.                 end = Math.min('z' + 1, end);
  289.             }
  290.         }

  291.         final int zeroDigitAscii = 48;
  292.         final int firstLetterAscii = 65;

  293.         if (chars == null && (numbers && end <= zeroDigitAscii || letters && end <= firstLetterAscii)) {
  294.             throw new IllegalArgumentException(
  295.                     "Parameter end (" + end + ") must be greater then (" + zeroDigitAscii + ") for generating digits "
  296.                             + "or greater then (" + firstLetterAscii + ") for generating letters.");
  297.         }

  298.         final StringBuilder builder = new StringBuilder(count);
  299.         final int gap = end - start;
  300.         final int gapBits = Integer.SIZE - Integer.numberOfLeadingZeros(gap);
  301.         // The size of the cache we use is an heuristic:
  302.         // about twice the number of bytes required if no rejection
  303.         // Ideally the cache size depends on multiple factor, including the cost of generating x bytes
  304.         // of randomness as well as the probability of rejection. It is however not easy to know
  305.         // those values programmatically for the general case.
  306.         final CachedRandomBits arb = new CachedRandomBits((count * gapBits + 3) / 5 + 10, random);

  307.         while (count-- != 0) {
  308.             // Generate a random value between start (included) and end (excluded)
  309.             final int randomValue = arb.nextBits(gapBits) + start;
  310.             // Rejection sampling if value too large
  311.             if (randomValue >= end) {
  312.                 count++;
  313.                 continue;
  314.             }

  315.             final int codePoint;
  316.             if (chars == null) {
  317.                 codePoint = randomValue;

  318.                 switch (Character.getType(codePoint)) {
  319.                 case Character.UNASSIGNED:
  320.                 case Character.PRIVATE_USE:
  321.                 case Character.SURROGATE:
  322.                     count++;
  323.                     continue;
  324.                 }

  325.             } else {
  326.                 codePoint = chars[randomValue];
  327.             }

  328.             final int numberOfChars = Character.charCount(codePoint);
  329.             if (count == 0 && numberOfChars > 1) {
  330.                 count++;
  331.                 continue;
  332.             }

  333.             if (letters && Character.isLetter(codePoint) || numbers && Character.isDigit(codePoint)
  334.                     || !letters && !numbers) {
  335.                 builder.appendCodePoint(codePoint);

  336.                 if (numberOfChars == 2) {
  337.                     count--;
  338.                 }

  339.             } else {
  340.                 count++;
  341.             }
  342.         }
  343.         return builder.toString();
  344.     }

  345.     /**
  346.      * Creates a random string whose length is the number of characters specified.
  347.      *
  348.      * <p>
  349.      * Characters will be chosen from the set of characters specified by the string, must not be empty. If null, the set
  350.      * of all characters is used.
  351.      * </p>
  352.      *
  353.      * @param count the length of random string to create
  354.      * @param chars the String containing the set of characters to use, may be null, but must not be empty
  355.      * @return the random string
  356.      * @throws IllegalArgumentException if {@code count} &lt; 0 or the string is empty.
  357.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  358.      */
  359.     @Deprecated
  360.     public static String random(final int count, final String chars) {
  361.         return secure().next(count, chars);
  362.     }

  363.     /**
  364.      * Creates a random string whose length is the number of characters specified.
  365.      *
  366.      * <p>
  367.      * Characters will be chosen from the set of Latin alphabetic characters (a-z, A-Z).
  368.      * </p>
  369.      *
  370.      * @param count the length of random string to create
  371.      * @return the random string
  372.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  373.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  374.      */
  375.     @Deprecated
  376.     public static String randomAlphabetic(final int count) {
  377.         return secure().nextAlphabetic(count);
  378.     }

  379.     /**
  380.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  381.      *
  382.      * <p>
  383.      * Characters will be chosen from the set of Latin alphabetic characters (a-z, A-Z).
  384.      * </p>
  385.      *
  386.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  387.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  388.      * @return the random string
  389.      * @since 3.5
  390.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  391.      */
  392.     @Deprecated
  393.     public static String randomAlphabetic(final int minLengthInclusive, final int maxLengthExclusive) {
  394.         return secure().nextAlphabetic(minLengthInclusive, maxLengthExclusive);
  395.     }

  396.     /**
  397.      * Creates a random string whose length is the number of characters specified.
  398.      *
  399.      * <p>
  400.      * Characters will be chosen from the set of Latin alphabetic characters (a-z, A-Z) and the digits 0-9.
  401.      * </p>
  402.      *
  403.      * @param count the length of random string to create
  404.      * @return the random string
  405.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  406.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  407.      */
  408.     @Deprecated
  409.     public static String randomAlphanumeric(final int count) {
  410.         return secure().nextAlphanumeric(count);
  411.     }

  412.     /**
  413.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  414.      *
  415.      * <p>
  416.      * Characters will be chosen from the set of Latin alphabetic characters (a-z, A-Z) and the digits 0-9.
  417.      * </p>
  418.      *
  419.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  420.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  421.      * @return the random string
  422.      * @since 3.5
  423.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  424.      */
  425.     @Deprecated
  426.     public static String randomAlphanumeric(final int minLengthInclusive, final int maxLengthExclusive) {
  427.         return secure().nextAlphanumeric(minLengthInclusive, maxLengthExclusive);
  428.     }

  429.     /**
  430.      * Creates a random string whose length is the number of characters specified.
  431.      *
  432.      * <p>
  433.      * Characters will be chosen from the set of characters whose ASCII value is between {@code 32} and {@code 126}
  434.      * (inclusive).
  435.      * </p>
  436.      *
  437.      * @param count the length of random string to create
  438.      * @return the random string
  439.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  440.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  441.      */
  442.     @Deprecated
  443.     public static String randomAscii(final int count) {
  444.         return secure().nextAscii(count);
  445.     }

  446.     /**
  447.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  448.      *
  449.      * <p>
  450.      * Characters will be chosen from the set of characters whose ASCII value is between {@code 32} and {@code 126}
  451.      * (inclusive).
  452.      * </p>
  453.      *
  454.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  455.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  456.      * @return the random string
  457.      * @since 3.5
  458.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  459.      */
  460.     @Deprecated
  461.     public static String randomAscii(final int minLengthInclusive, final int maxLengthExclusive) {
  462.         return secure().nextAscii(minLengthInclusive, maxLengthExclusive);
  463.     }

  464.     /**
  465.      * Creates a random string whose length is the number of characters specified.
  466.      *
  467.      * <p>
  468.      * Characters will be chosen from the set of characters which match the POSIX [:graph:] regular expression character
  469.      * class. This class contains all visible ASCII characters (i.e. anything except spaces and control characters).
  470.      * </p>
  471.      *
  472.      * @param count the length of random string to create
  473.      * @return the random string
  474.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  475.      * @since 3.5
  476.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  477.      */
  478.     @Deprecated
  479.     public static String randomGraph(final int count) {
  480.         return secure().nextGraph(count);
  481.     }

  482.     /**
  483.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  484.      *
  485.      * <p>
  486.      * Characters will be chosen from the set of \p{Graph} characters.
  487.      * </p>
  488.      *
  489.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  490.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  491.      * @return the random string
  492.      * @since 3.5
  493.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  494.      */
  495.     @Deprecated
  496.     public static String randomGraph(final int minLengthInclusive, final int maxLengthExclusive) {
  497.         return secure().nextGraph(minLengthInclusive, maxLengthExclusive);
  498.     }

  499.     /**
  500.      * Creates a random string whose length is the number of characters specified.
  501.      *
  502.      * <p>
  503.      * Characters will be chosen from the set of numeric characters.
  504.      * </p>
  505.      *
  506.      * @param count the length of random string to create
  507.      * @return the random string
  508.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  509.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  510.      */
  511.     @Deprecated
  512.     public static String randomNumeric(final int count) {
  513.         return secure().nextNumeric(count);
  514.     }

  515.     /**
  516.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  517.      *
  518.      * <p>
  519.      * Characters will be chosen from the set of \p{Digit} characters.
  520.      * </p>
  521.      *
  522.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  523.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  524.      * @return the random string
  525.      * @since 3.5
  526.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  527.      */
  528.     @Deprecated
  529.     public static String randomNumeric(final int minLengthInclusive, final int maxLengthExclusive) {
  530.         return secure().nextNumeric(minLengthInclusive, maxLengthExclusive);
  531.     }

  532.     /**
  533.      * Creates a random string whose length is the number of characters specified.
  534.      *
  535.      * <p>
  536.      * Characters will be chosen from the set of characters which match the POSIX [:print:] regular expression character
  537.      * class. This class includes all visible ASCII characters and spaces (i.e. anything except control characters).
  538.      * </p>
  539.      *
  540.      * @param count the length of random string to create
  541.      * @return the random string
  542.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  543.      * @since 3.5
  544.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  545.      */
  546.     @Deprecated
  547.     public static String randomPrint(final int count) {
  548.         return secure().nextPrint(count);
  549.     }

  550.     /**
  551.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  552.      *
  553.      * <p>
  554.      * Characters will be chosen from the set of \p{Print} characters.
  555.      * </p>
  556.      *
  557.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  558.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  559.      * @return the random string
  560.      * @since 3.5
  561.      * @deprecated Use {@link #secure()}, {@link #secureStrong()},or {@link #insecure()}.
  562.      */
  563.     @Deprecated
  564.     public static String randomPrint(final int minLengthInclusive, final int maxLengthExclusive) {
  565.         return secure().nextPrint(minLengthInclusive, maxLengthExclusive);
  566.     }

  567.     /**
  568.      * Gets the singleton instance based on {@link SecureRandom#SecureRandom()} which uses a secure random number generator (RNG) implementing the default
  569.      * random number algorithm.
  570.      * <p>
  571.      * The method {@link SecureRandom#SecureRandom()} is called on-demand.
  572.      * </p>
  573.      *
  574.      * @return the singleton instance based on {@link SecureRandom#SecureRandom()}.
  575.      * @see SecureRandom#SecureRandom()
  576.      * @since 3.16.0
  577.      */
  578.     public static RandomStringUtils secure() {
  579.         return SECURE;
  580.     }

  581.     /**
  582.      * Gets the singleton instance based on {@link SecureRandom#getInstanceStrong()} which uses an algorithms/providers
  583.      * specified in the {@code securerandom.strongAlgorithms} {@link Security} property.
  584.      * <p>
  585.      * The method {@link SecureRandom#getInstanceStrong()} is called on-demand.
  586.      * </p>
  587.      *
  588.      * @return the singleton instance based on {@link SecureRandom#getInstanceStrong()}.
  589.      * @see SecureRandom#getInstanceStrong()
  590.      * @since 3.17.0
  591.      */
  592.     public static RandomStringUtils secureStrong() {
  593.         return SECURE_STRONG;
  594.     }

  595.     private final Supplier<RandomUtils> random;

  596.     /**
  597.      * {@link RandomStringUtils} instances should NOT be constructed in standard programming. Instead, the class should
  598.      * be used as {@code RandomStringUtils.random(5);}.
  599.      *
  600.      * <p>
  601.      * This constructor is public to permit tools that require a JavaBean instance to operate.
  602.      * </p>
  603.      *
  604.      * @deprecated TODO Make private in 4.0.
  605.      */
  606.     @Deprecated
  607.     public RandomStringUtils() {
  608.         this(SECURE_SUPPLIER);
  609.     }

  610.     private RandomStringUtils(final Supplier<RandomUtils> random) {
  611.         this.random = random;
  612.     }

  613.     /**
  614.      * Creates a random string whose length is the number of characters specified.
  615.      *
  616.      * <p>
  617.      * Characters will be chosen from the set of all characters.
  618.      * </p>
  619.      *
  620.      * @param count the length of random string to create
  621.      * @return the random string
  622.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  623.      * @since 3.16.0
  624.      */
  625.     public String next(final int count) {
  626.         return next(count, false, false);
  627.     }

  628.     /**
  629.      * Creates a random string whose length is the number of characters specified.
  630.      *
  631.      * <p>
  632.      * Characters will be chosen from the set of alpha-numeric characters as indicated by the arguments.
  633.      * </p>
  634.      *
  635.      * @param count   the length of random string to create
  636.      * @param letters if {@code true}, generated string may include alphabetic characters
  637.      * @param numbers if {@code true}, generated string may include numeric characters
  638.      * @return the random string
  639.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  640.      * @since 3.16.0
  641.      */
  642.     public String next(final int count, final boolean letters, final boolean numbers) {
  643.         return next(count, 0, 0, letters, numbers);
  644.     }

  645.     /**
  646.      * Creates a random string whose length is the number of characters specified.
  647.      *
  648.      * <p>
  649.      * Characters will be chosen from the set of characters specified.
  650.      * </p>
  651.      *
  652.      * @param count the length of random string to create
  653.      * @param chars the character array containing the set of characters to use, may be null
  654.      * @return the random string
  655.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  656.      * @since 3.16.0
  657.      */
  658.     public String next(final int count, final char... chars) {
  659.         if (chars == null) {
  660.             return random(count, 0, 0, false, false, null, random());
  661.         }
  662.         return random(count, 0, chars.length, false, false, chars, random());
  663.     }

  664.     /**
  665.      * Creates a random string whose length is the number of characters specified.
  666.      *
  667.      * <p>
  668.      * Characters will be chosen from the set of alpha-numeric characters as indicated by the arguments.
  669.      * </p>
  670.      *
  671.      * @param count   the length of random string to create
  672.      * @param start   the position in set of chars to start at
  673.      * @param end     the position in set of chars to end before
  674.      * @param letters if {@code true}, generated string may include alphabetic characters
  675.      * @param numbers if {@code true}, generated string may include numeric characters
  676.      * @return the random string
  677.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  678.      * @since 3.16.0
  679.      */
  680.     public String next(final int count, final int start, final int end, final boolean letters, final boolean numbers) {
  681.         return random(count, start, end, letters, numbers, null, random());
  682.     }

  683.     /**
  684.      * Creates a random string based on a variety of options, using default source of randomness.
  685.      *
  686.      * <p>
  687.      * This method has exactly the same semantics as {@link #random(int,int,int,boolean,boolean,char[],Random)}, but
  688.      * instead of using an externally supplied source of randomness, it uses the internal static {@link Random}
  689.      * instance.
  690.      * </p>
  691.      *
  692.      * @param count   the length of random string to create
  693.      * @param start   the position in set of chars to start at
  694.      * @param end     the position in set of chars to end before
  695.      * @param letters if {@code true}, generated string may include alphabetic characters
  696.      * @param numbers if {@code true}, generated string may include numeric characters
  697.      * @param chars   the set of chars to choose randoms from. If {@code null}, then it will use the set of all chars.
  698.      * @return the random string
  699.      * @throws ArrayIndexOutOfBoundsException if there are not {@code (end - start) + 1} characters in the set array.
  700.      * @throws IllegalArgumentException       if {@code count} &lt; 0.
  701.      */
  702.     public String next(final int count, final int start, final int end, final boolean letters, final boolean numbers,
  703.             final char... chars) {
  704.         return random(count, start, end, letters, numbers, chars, random());
  705.     }

  706.     /**
  707.      * Creates a random string whose length is the number of characters specified.
  708.      *
  709.      * <p>
  710.      * Characters will be chosen from the set of characters specified by the string, must not be empty. If null, the set
  711.      * of all characters is used.
  712.      * </p>
  713.      *
  714.      * @param count the length of random string to create
  715.      * @param chars the String containing the set of characters to use, may be null, but must not be empty
  716.      * @return the random string
  717.      * @throws IllegalArgumentException if {@code count} &lt; 0 or the string is empty.
  718.      * @since 3.16.0
  719.      */
  720.     public String next(final int count, final String chars) {
  721.         if (chars == null) {
  722.             return random(count, 0, 0, false, false, null, random());
  723.         }
  724.         return next(count, chars.toCharArray());
  725.     }

  726.     /**
  727.      * Creates a random string whose length is the number of characters specified.
  728.      *
  729.      * <p>
  730.      * Characters will be chosen from the set of Latin alphabetic characters (a-z, A-Z).
  731.      * </p>
  732.      *
  733.      * @param count the length of random string to create
  734.      * @return the random string
  735.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  736.      */
  737.     public String nextAlphabetic(final int count) {
  738.         return next(count, true, false);
  739.     }

  740.     /**
  741.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  742.      *
  743.      * <p>
  744.      * Characters will be chosen from the set of Latin alphabetic characters (a-z, A-Z).
  745.      * </p>
  746.      *
  747.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  748.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  749.      * @return the random string
  750.      * @since 3.5
  751.      */
  752.     public String nextAlphabetic(final int minLengthInclusive, final int maxLengthExclusive) {
  753.         return nextAlphabetic(randomUtils().randomInt(minLengthInclusive, maxLengthExclusive));
  754.     }

  755.     /**
  756.      * Creates a random string whose length is the number of characters specified.
  757.      *
  758.      * <p>
  759.      * Characters will be chosen from the set of Latin alphabetic characters (a-z, A-Z) and the digits 0-9.
  760.      * </p>
  761.      *
  762.      * @param count the length of random string to create
  763.      * @return the random string
  764.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  765.      */
  766.     public String nextAlphanumeric(final int count) {
  767.         return next(count, true, true);
  768.     }

  769.     /**
  770.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  771.      *
  772.      * <p>
  773.      * Characters will be chosen from the set of Latin alphabetic characters (a-z, A-Z) and the digits 0-9.
  774.      * </p>
  775.      *
  776.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  777.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  778.      * @return the random string
  779.      * @since 3.5
  780.      */
  781.     public String nextAlphanumeric(final int minLengthInclusive, final int maxLengthExclusive) {
  782.         return nextAlphanumeric(randomUtils().randomInt(minLengthInclusive, maxLengthExclusive));
  783.     }

  784.     /**
  785.      * Creates a random string whose length is the number of characters specified.
  786.      *
  787.      * <p>
  788.      * Characters will be chosen from the set of characters whose ASCII value is between {@code 32} and {@code 126}
  789.      * (inclusive).
  790.      * </p>
  791.      *
  792.      * @param count the length of random string to create
  793.      * @return the random string
  794.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  795.      */
  796.     public String nextAscii(final int count) {
  797.         return next(count, 32, 127, false, false);
  798.     }

  799.     /**
  800.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  801.      *
  802.      * <p>
  803.      * Characters will be chosen from the set of characters whose ASCII value is between {@code 32} and {@code 126}
  804.      * (inclusive).
  805.      * </p>
  806.      *
  807.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  808.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  809.      * @return the random string
  810.      * @since 3.5
  811.      */
  812.     public String nextAscii(final int minLengthInclusive, final int maxLengthExclusive) {
  813.         return nextAscii(randomUtils().randomInt(minLengthInclusive, maxLengthExclusive));
  814.     }

  815.     /**
  816.      * Creates a random string whose length is the number of characters specified.
  817.      *
  818.      * <p>
  819.      * Characters will be chosen from the set of characters which match the POSIX [:graph:] regular expression character
  820.      * class. This class contains all visible ASCII characters (i.e. anything except spaces and control characters).
  821.      * </p>
  822.      *
  823.      * @param count the length of random string to create
  824.      * @return the random string
  825.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  826.      * @since 3.5
  827.      */
  828.     public String nextGraph(final int count) {
  829.         return next(count, 33, 126, false, false);
  830.     }

  831.     /**
  832.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  833.      *
  834.      * <p>
  835.      * Characters will be chosen from the set of \p{Graph} characters.
  836.      * </p>
  837.      *
  838.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  839.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  840.      * @return the random string
  841.      * @since 3.5
  842.      */
  843.     public String nextGraph(final int minLengthInclusive, final int maxLengthExclusive) {
  844.         return nextGraph(randomUtils().randomInt(minLengthInclusive, maxLengthExclusive));
  845.     }

  846.     /**
  847.      * Creates a random string whose length is the number of characters specified.
  848.      *
  849.      * <p>
  850.      * Characters will be chosen from the set of numeric characters.
  851.      * </p>
  852.      *
  853.      * @param count the length of random string to create
  854.      * @return the random string
  855.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  856.      */
  857.     public String nextNumeric(final int count) {
  858.         return next(count, false, true);
  859.     }

  860.     /**
  861.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  862.      *
  863.      * <p>
  864.      * Characters will be chosen from the set of \p{Digit} characters.
  865.      * </p>
  866.      *
  867.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  868.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  869.      * @return the random string
  870.      * @since 3.5
  871.      */
  872.     public String nextNumeric(final int minLengthInclusive, final int maxLengthExclusive) {
  873.         return nextNumeric(randomUtils().randomInt(minLengthInclusive, maxLengthExclusive));
  874.     }

  875.     /**
  876.      * Creates a random string whose length is the number of characters specified.
  877.      *
  878.      * <p>
  879.      * Characters will be chosen from the set of characters which match the POSIX [:print:] regular expression character
  880.      * class. This class includes all visible ASCII characters and spaces (i.e. anything except control characters).
  881.      * </p>
  882.      *
  883.      * @param count the length of random string to create
  884.      * @return the random string
  885.      * @throws IllegalArgumentException if {@code count} &lt; 0.
  886.      * @since 3.5
  887.      * @since 3.16.0
  888.      */
  889.     public String nextPrint(final int count) {
  890.         return next(count, 32, 126, false, false);
  891.     }

  892.     /**
  893.      * Creates a random string whose length is between the inclusive minimum and the exclusive maximum.
  894.      *
  895.      * <p>
  896.      * Characters will be chosen from the set of \p{Print} characters.
  897.      * </p>
  898.      *
  899.      * @param minLengthInclusive the inclusive minimum length of the string to generate
  900.      * @param maxLengthExclusive the exclusive maximum length of the string to generate
  901.      * @return the random string
  902.      * @since 3.16.0
  903.      */
  904.     public String nextPrint(final int minLengthInclusive, final int maxLengthExclusive) {
  905.         return nextPrint(randomUtils().randomInt(minLengthInclusive, maxLengthExclusive));
  906.     }

  907.     /**
  908.      * Gets the Random.
  909.      *
  910.      * @return the Random.
  911.      */
  912.     private Random random() {
  913.         return randomUtils().random();
  914.     }

  915.     /**
  916.      * Gets the RandomUtils.
  917.      *
  918.      * @return the RandomUtils.
  919.      */
  920.     private RandomUtils randomUtils() {
  921.         return random.get();
  922.     }

  923.     @Override
  924.     public String toString() {
  925.         return "RandomStringUtils [random=" + random() + "]";
  926.     }

  927. }