InetAddressValidator.java
- /*
- * Licensed to the Apache Software Foundation (ASF) under one or more
- * contributor license agreements. See the NOTICE file distributed with
- * this work for additional information regarding copyright ownership.
- * The ASF licenses this file to You under the Apache License, Version 2.0
- * (the "License"); you may not use this file except in compliance with
- * the License. You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- package org.apache.commons.validator.routines;
- import java.io.Serializable;
- import java.util.ArrayList;
- import java.util.Arrays;
- import java.util.List;
- import java.util.regex.Pattern;
- import org.apache.commons.validator.GenericValidator;
- /**
- * <p><b>InetAddress</b> validation and conversion routines (<code>java.net.InetAddress</code>).</p>
- *
- * <p>This class provides methods to validate a candidate IP address.
- *
- * <p>
- * This class is a Singleton; you can retrieve the instance via the {@link #getInstance()} method.
- * </p>
- *
- * @since 1.4
- */
- public class InetAddressValidator implements Serializable {
- private static final int MAX_BYTE = 128;
- private static final int IPV4_MAX_OCTET_VALUE = 255;
- private static final int MAX_UNSIGNED_SHORT = 0xffff;
- private static final int BASE_16 = 16;
- private static final long serialVersionUID = -919201640201914789L;
- private static final String IPV4_REGEX =
- "^(\\d{1,3})\\.(\\d{1,3})\\.(\\d{1,3})\\.(\\d{1,3})$";
- // Max number of hex groups (separated by :) in an IPV6 address
- private static final int IPV6_MAX_HEX_GROUPS = 8;
- // Max hex digits in each IPv6 group
- private static final int IPV6_MAX_HEX_DIGITS_PER_GROUP = 4;
- /**
- * Singleton instance of this class.
- */
- private static final InetAddressValidator VALIDATOR = new InetAddressValidator();
- private static final Pattern DIGITS_PATTERN = Pattern.compile("\\d{1,3}");
- private static final Pattern ID_CHECK_PATTERN = Pattern.compile("[^\\s/%]+");
- /**
- * Returns the singleton instance of this validator.
- * @return the singleton instance of this validator
- */
- public static InetAddressValidator getInstance() {
- return VALIDATOR;
- }
- /** IPv4 RegexValidator */
- private final RegexValidator ipv4Validator = new RegexValidator(IPV4_REGEX);
- /**
- * Checks if the specified string is a valid IPv4 or IPv6 address.
- * @param inetAddress the string to validate
- * @return true if the string validates as an IP address
- */
- public boolean isValid(final String inetAddress) {
- return isValidInet4Address(inetAddress) || isValidInet6Address(inetAddress);
- }
- /**
- * Validates an IPv4 address. Returns true if valid.
- * @param inet4Address the IPv4 address to validate
- * @return true if the argument contains a valid IPv4 address
- */
- public boolean isValidInet4Address(final String inet4Address) {
- // verify that address conforms to generic IPv4 format
- final String[] groups = ipv4Validator.match(inet4Address);
- if (groups == null) {
- return false;
- }
- // verify that address subgroups are legal
- for (final String ipSegment : groups) {
- if (GenericValidator.isBlankOrNull(ipSegment)) {
- return false;
- }
- int iIpSegment = 0;
- try {
- iIpSegment = Integer.parseInt(ipSegment);
- } catch (final NumberFormatException e) {
- return false;
- }
- if (iIpSegment > IPV4_MAX_OCTET_VALUE) {
- return false;
- }
- if (ipSegment.length() > 1 && ipSegment.startsWith("0")) {
- return false;
- }
- }
- return true;
- }
- /**
- * Validates an IPv6 address. Returns true if valid.
- * @param inet6Address the IPv6 address to validate
- * @return true if the argument contains a valid IPv6 address
- *
- * @since 1.4.1
- */
- public boolean isValidInet6Address(String inet6Address) {
- String[] parts;
- // remove prefix size. This will appear after the zone id (if any)
- parts = inet6Address.split("/", -1);
- if (parts.length > 2) {
- return false; // can only have one prefix specifier
- }
- if (parts.length == 2) {
- if (!DIGITS_PATTERN.matcher(parts[1]).matches()) {
- return false; // not a valid number
- }
- final int bits = Integer.parseInt(parts[1]); // cannot fail because of RE check
- if (bits < 0 || bits > MAX_BYTE) {
- return false; // out of range
- }
- }
- // remove zone-id
- parts = parts[0].split("%", -1);
- if (parts.length > 2) {
- return false;
- }
- // The id syntax is implementation independent, but it presumably cannot allow:
- // whitespace, '/' or '%'
- if (parts.length == 2 && !ID_CHECK_PATTERN.matcher(parts[1]).matches()) {
- return false; // invalid id
- }
- inet6Address = parts[0];
- final boolean containsCompressedZeroes = inet6Address.contains("::");
- if (containsCompressedZeroes && inet6Address.indexOf("::") != inet6Address.lastIndexOf("::")) {
- return false;
- }
- if (inet6Address.startsWith(":") && !inet6Address.startsWith("::")
- || inet6Address.endsWith(":") && !inet6Address.endsWith("::")) {
- return false;
- }
- String[] octets = inet6Address.split(":");
- if (containsCompressedZeroes) {
- final List<String> octetList = new ArrayList<>(Arrays.asList(octets));
- if (inet6Address.endsWith("::")) {
- // String.split() drops ending empty segments
- octetList.add("");
- } else if (inet6Address.startsWith("::") && !octetList.isEmpty()) {
- octetList.remove(0);
- }
- octets = octetList.toArray(new String[0]);
- }
- if (octets.length > IPV6_MAX_HEX_GROUPS) {
- return false;
- }
- int validOctets = 0;
- int emptyOctets = 0; // consecutive empty chunks
- for (int index = 0; index < octets.length; index++) {
- final String octet = octets[index];
- if (GenericValidator.isBlankOrNull(octet)) {
- emptyOctets++;
- if (emptyOctets > 1) {
- return false;
- }
- } else {
- emptyOctets = 0;
- // Is last chunk an IPv4 address?
- if (index == octets.length - 1 && octet.contains(".")) {
- if (!isValidInet4Address(octet)) {
- return false;
- }
- validOctets += 2;
- continue;
- }
- if (octet.length() > IPV6_MAX_HEX_DIGITS_PER_GROUP) {
- return false;
- }
- int octetInt = 0;
- try {
- octetInt = Integer.parseInt(octet, BASE_16);
- } catch (final NumberFormatException e) {
- return false;
- }
- if (octetInt < 0 || octetInt > MAX_UNSIGNED_SHORT) {
- return false;
- }
- }
- validOctets++;
- }
- if (validOctets > IPV6_MAX_HEX_GROUPS || validOctets < IPV6_MAX_HEX_GROUPS && !containsCompressedZeroes) {
- return false;
- }
- return true;
- }
- }