JavaVersion.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 org.apache.commons.lang3.math.NumberUtils;

  19. /**
  20.  * An enum representing all the versions of the Java specification.
  21.  * This is intended to mirror available values from the
  22.  * <em>java.specification.version</em> System property.
  23.  *
  24.  * @since 3.0
  25.  */
  26. public enum JavaVersion {

  27.     /**
  28.      * The Java version reported by Android. This is not an official Java version number.
  29.      */
  30.     JAVA_0_9(1.5f, "0.9"),

  31.     /**
  32.      * Java 1.1.
  33.      */
  34.     JAVA_1_1(1.1f, "1.1"),

  35.     /**
  36.      * Java 1.2.
  37.      */
  38.     JAVA_1_2(1.2f, "1.2"),

  39.     /**
  40.      * Java 1.3.
  41.      */
  42.     JAVA_1_3(1.3f, "1.3"),

  43.     /**
  44.      * Java 1.4.
  45.      */
  46.     JAVA_1_4(1.4f, "1.4"),

  47.     /**
  48.      * Java 1.5.
  49.      */
  50.     JAVA_1_5(1.5f, "1.5"),

  51.     /**
  52.      * Java 1.6.
  53.      */
  54.     JAVA_1_6(1.6f, "1.6"),

  55.     /**
  56.      * Java 1.7.
  57.      */
  58.     JAVA_1_7(1.7f, "1.7"),

  59.     /**
  60.      * Java 1.8.
  61.      */
  62.     JAVA_1_8(1.8f, "1.8"),

  63.     /**
  64.      * Java 1.9.
  65.      *
  66.      * @deprecated As of release 3.5, replaced by {@link #JAVA_9}
  67.      */
  68.     @Deprecated
  69.     JAVA_1_9(9.0f, "9"),

  70.     /**
  71.      * Java 9.
  72.      *
  73.      * @since 3.5
  74.      */
  75.     JAVA_9(9.0f, "9"),

  76.     /**
  77.      * Java 10.
  78.      *
  79.      * @since 3.7
  80.      */
  81.     JAVA_10(10.0f, "10"),

  82.     /**
  83.      * Java 11.
  84.      *
  85.      * @since 3.8
  86.      */
  87.     JAVA_11(11.0f, "11"),

  88.     /**
  89.      * Java 12.
  90.      *
  91.      * @since 3.9
  92.      */
  93.     JAVA_12(12.0f, "12"),

  94.     /**
  95.      * Java 13.
  96.      *
  97.      * @since 3.9
  98.      */
  99.     JAVA_13(13.0f, "13"),

  100.     /**
  101.      * Java 14.
  102.      *
  103.      * @since 3.11
  104.      */
  105.     JAVA_14(14.0f, "14"),

  106.     /**
  107.      * Java 15.
  108.      *
  109.      * @since 3.11
  110.      */
  111.     JAVA_15(15.0f, "15"),

  112.     /**
  113.      * Java 16.
  114.      *
  115.      * @since 3.11
  116.      */
  117.     JAVA_16(16.0f, "16"),

  118.     /**
  119.      * Java 17.
  120.      *
  121.      * @since 3.12.0
  122.      */
  123.     JAVA_17(17.0f, "17"),

  124.     /**
  125.      * Java 18.
  126.      *
  127.      * @since 3.13.0
  128.      */
  129.     JAVA_18(18.0f, "18"),

  130.     /**
  131.      * Java 19.
  132.      *
  133.      * @since 3.13.0
  134.      */
  135.     JAVA_19(19.0f, "19"),

  136.     /**
  137.      * Java 20.
  138.      *
  139.      * @since 3.13.0
  140.      */
  141.     JAVA_20(20, "20"),

  142.     /**
  143.      * Java 21.
  144.      *
  145.      * @since 3.13.0
  146.      */
  147.     JAVA_21(21, "21"),

  148.     /**
  149.      * Java 22.
  150.      *
  151.      * @since 3.15.0
  152.      */
  153.     JAVA_22(22, "22"),

  154.     /**
  155.      * The most recent Java version. Mainly introduced to avoid to break when a new version of Java is used.
  156.      */
  157.     JAVA_RECENT(maxVersion(), Float.toString(maxVersion()));

  158.     /**
  159.      * The regex to split version strings.
  160.      */
  161.     private static final String VERSION_SPLIT_REGEX = "\\.";

  162.     /**
  163.      * Transforms the given string with a Java version number to the
  164.      * corresponding constant of this enumeration class. This method is used
  165.      * internally.
  166.      *
  167.      * @param versionStr the Java version as string
  168.      * @return the corresponding enumeration constant or <b>null</b> if the
  169.      * version is unknown
  170.      */
  171.     static JavaVersion get(final String versionStr) {
  172.         if (versionStr == null) {
  173.             return null;
  174.         }
  175.         switch (versionStr) {
  176.         case "0.9":
  177.             return JAVA_0_9;
  178.         case "1.1":
  179.             return JAVA_1_1;
  180.         case "1.2":
  181.             return JAVA_1_2;
  182.         case "1.3":
  183.             return JAVA_1_3;
  184.         case "1.4":
  185.             return JAVA_1_4;
  186.         case "1.5":
  187.             return JAVA_1_5;
  188.         case "1.6":
  189.             return JAVA_1_6;
  190.         case "1.7":
  191.             return JAVA_1_7;
  192.         case "1.8":
  193.             return JAVA_1_8;
  194.         case "9":
  195.             return JAVA_9;
  196.         case "10":
  197.             return JAVA_10;
  198.         case "11":
  199.             return JAVA_11;
  200.         case "12":
  201.             return JAVA_12;
  202.         case "13":
  203.             return JAVA_13;
  204.         case "14":
  205.             return JAVA_14;
  206.         case "15":
  207.             return JAVA_15;
  208.         case "16":
  209.             return JAVA_16;
  210.         case "17":
  211.             return JAVA_17;
  212.         case "18":
  213.             return JAVA_18;
  214.         case "19":
  215.             return JAVA_19;
  216.         case "20":
  217.             return JAVA_20;
  218.         case "21":
  219.             return JAVA_21;
  220.         case "22":
  221.             return JAVA_22;
  222.         default:
  223.             final float v = toFloatVersion(versionStr);
  224.             if (v - 1. < 1.) { // then we need to check decimals > .9
  225.                 final int firstComma = Math.max(versionStr.indexOf('.'), versionStr.indexOf(','));
  226.                 final int end = Math.max(versionStr.length(), versionStr.indexOf(',', firstComma));
  227.                 if (Float.parseFloat(versionStr.substring(firstComma + 1, end)) > .9f) {
  228.                     return JAVA_RECENT;
  229.                 }
  230.             } else if (v > 10) {
  231.                 return JAVA_RECENT;
  232.             }
  233.             return null;
  234.         }
  235.     }

  236.     /**
  237.      * Transforms the given string with a Java version number to the
  238.      * corresponding constant of this enumeration class. This method is used
  239.      * internally.
  240.      *
  241.      * @param versionStr the Java version as string
  242.      * @return the corresponding enumeration constant or <b>null</b> if the
  243.      * version is unknown
  244.      */
  245.     static JavaVersion getJavaVersion(final String versionStr) {
  246.         return get(versionStr);
  247.     }

  248.     /**
  249.      * Gets the Java Version from the system or 99.0 if the {@code java.specification.version} system property is not set.
  250.      *
  251.      * @return the value of {@code java.specification.version} system property or 99.0 if it is not set.
  252.      */
  253.     private static float maxVersion() {
  254.         final float v = toFloatVersion(SystemProperties.getJavaSpecificationVersion("99.0"));
  255.         return v > 0 ? v : 99f;
  256.     }

  257.     static String[] split(final String value) {
  258.         return value.split(VERSION_SPLIT_REGEX);
  259.     }

  260.     /**
  261.      * Parses a float value from a String.
  262.      *
  263.      * @param value the String to parse.
  264.      * @return the float value represented by the string or -1 if the given String can not be parsed.
  265.      */
  266.     private static float toFloatVersion(final String value) {
  267.         final int defaultReturnValue = -1;
  268.         if (!value.contains(".")) {
  269.             return NumberUtils.toFloat(value, defaultReturnValue);
  270.         }
  271.         final String[] toParse = split(value);
  272.         if (toParse.length >= 2) {
  273.             return NumberUtils.toFloat(toParse[0] + '.' + toParse[1], defaultReturnValue);
  274.         }
  275.         return defaultReturnValue;
  276.     }

  277.     /**
  278.      * The float value.
  279.      */
  280.     private final float value;

  281.     /**
  282.      * The standard name.
  283.      */
  284.     private final String name;

  285.     /**
  286.      * Constructs a new instance.
  287.      *
  288.      * @param value  the float value
  289.      * @param name  the standard name, not null
  290.      */
  291.     JavaVersion(final float value, final String name) {
  292.         this.value = value;
  293.         this.name = name;
  294.     }

  295.     /**
  296.      * Tests whether this version of Java is at least the version of Java passed in.
  297.      *
  298.      * <p>For example:<br>
  299.      *  {@code myVersion.atLeast(JavaVersion.JAVA_1_4)}</p>
  300.      *
  301.      * @param requiredVersion  the version to check against, not null
  302.      * @return true if this version is equal to or greater than the specified version
  303.      */
  304.     public boolean atLeast(final JavaVersion requiredVersion) {
  305.         return this.value >= requiredVersion.value;
  306.     }

  307.     /**
  308.      * Tests whether this version of Java is at most the version of Java passed in.
  309.      *
  310.      * <p>For example:<br>
  311.      *  {@code myVersion.atMost(JavaVersion.JAVA_1_4)}</p>
  312.      *
  313.      * @param requiredVersion  the version to check against, not null
  314.      * @return true if this version is equal to or greater than the specified version
  315.      * @since 3.9
  316.      */
  317.     public boolean atMost(final JavaVersion requiredVersion) {
  318.         return this.value <= requiredVersion.value;
  319.     }

  320.     /**
  321.      * The string value is overridden to return the standard name.
  322.      *
  323.      * <p>For example, {@code "1.5"}.</p>
  324.      *
  325.      * @return the name, not null
  326.      */
  327.     @Override
  328.     public String toString() {
  329.         return name;
  330.     }
  331. }