-
Zoey76 authored8c677ebe
IPSubnet.java 3.55 KiB
/*
* Copyright © 2004-2020 L2J Server
*
* This file is part of L2J Server.
*
* L2J Server is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* L2J Server is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package com.l2jserver.loginserver.util;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.Arrays;
public class IPSubnet {
final byte[] _addr;
final byte[] _mask;
final boolean _isIPv4;
public IPSubnet(String input) throws UnknownHostException, NumberFormatException, ArrayIndexOutOfBoundsException {
int idx = input.indexOf("/");
if (idx > 0) {
_addr = InetAddress.getByName(input.substring(0, idx)).getAddress();
_mask = getMask(Integer.parseInt(input.substring(idx + 1)), _addr.length);
_isIPv4 = _addr.length == 4;
if (!applyMask(_addr)) {
throw new UnknownHostException(input);
}
} else {
_addr = InetAddress.getByName(input).getAddress();
_mask = getMask(_addr.length * 8, _addr.length); // host, no need to check mask
_isIPv4 = _addr.length == 4;
}
}
public IPSubnet(InetAddress addr, int mask) throws UnknownHostException {
_addr = addr.getAddress();
_isIPv4 = _addr.length == 4;
_mask = getMask(mask, _addr.length);
if (!applyMask(_addr)) {
throw new UnknownHostException(addr + "/" + mask);
}
}
public byte[] getAddress() {
return _addr;
}
public boolean applyMask(byte[] addr) {
// V4 vs V4 or V6 vs V6 checks
if (_isIPv4 == (addr.length == 4)) {
for (int i = 0; i < _addr.length; i++) {
if ((addr[i] & _mask[i]) != _addr[i]) {
return false;
}
}
} else {
// check for embedded v4 in v6 addr (not done !)
if (_isIPv4) {
// my V4 vs V6
for (int i = 0; i < _addr.length; i++) {
if ((addr[i + 12] & _mask[i]) != _addr[i]) {
return false;
}
}
} else {
// my V6 vs V4
for (int i = 0; i < _addr.length; i++) {
if ((addr[i] & _mask[i + 12]) != _addr[i + 12]) {
return false;
}
}
}
}
return true;
}
@Override
public String toString() {
int size = 0;
for (byte element : _mask) {
size += Integer.bitCount((element & 0xFF));
}
try {
return InetAddress.getByAddress(_addr) + "/" + size;
} catch (UnknownHostException e) {
return "Invalid";
}
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o instanceof IPSubnet) {
return applyMask(((IPSubnet) o).getAddress());
} else if (o instanceof InetAddress) {
return applyMask(((InetAddress) o).getAddress());
}
return false;
}
private static byte[] getMask(int n, int maxLength) throws UnknownHostException {
if ((n > (maxLength << 3)) || (n < 0)) {
throw new UnknownHostException("Invalid netmask: " + n);
}
final byte[] result = new byte[maxLength];
Arrays.fill(result, (byte) 0xFF);
for (int i = (maxLength << 3) - 1; i >= n; i--) {
result[i >> 3] = (byte) (result[i >> 3] << 1);
}
return result;
}
}