191 lines
6.0 KiB
Java
191 lines
6.0 KiB
Java
import java.net.Inet6Address;
|
|
import java.net.InetAddress;
|
|
import java.net.UnknownHostException;
|
|
import java.util.regex.Pattern;
|
|
|
|
import javax.xml.bind.DatatypeConverter;
|
|
|
|
/**
|
|
* Parses ipv4 and ipv6 addresses. Emits each described IP address as a
|
|
* hexadecimal integer representing the address, the address space, and the port
|
|
* number specified, if any.
|
|
*/
|
|
public class IPParser {
|
|
/*
|
|
* Using regex to ensure that the address is a valid one. This allows for
|
|
* separating by format and ensures that the operations done on a format
|
|
* will be valid.
|
|
*/
|
|
// 0.0.0.0-255.255.255.255
|
|
private final String ipv4segment =
|
|
"(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])";
|
|
|
|
// 0-65535
|
|
private final String portsegment =
|
|
":(?:6553[0-5]|655[0-2][0-9]|65[0-4][0-9]{2}|"
|
|
+ "6[0-4][0-9]{3}|[1-5][0-9]{4}|[1-9][0-9]{1,3}|[0-9])";
|
|
private final String ipv4address = "(" + ipv4segment + "\\.){3,3}"
|
|
+ ipv4segment;
|
|
private final String ipv4addressWithPort = ipv4address + portsegment + "?";
|
|
private final String ipv6segment = "[a-fA-F0-9]{1,4}";
|
|
private final String ipv6address = "(" +
|
|
// 1:2:3:4:5:6:7:8
|
|
"(" + ipv6segment + ":){7,7}" + ipv6segment + "|" +
|
|
// 1::, 1:2:3:4:5:6:7::
|
|
"(" + ipv6segment + ":){1,7}:|" +
|
|
// 1::8, 1:2:3:4:5:6::8, 1:2:3:4:5:6::8
|
|
"(" + ipv6segment + ":){1,6}:" + ipv6segment + "|" +
|
|
// 1::7:8, 1:2:3:4:5::7:8, 1:2:3:4:5::8
|
|
"(" + ipv6segment + ":){1,5}(:" + ipv6segment + "){1,2}|" +
|
|
// 1::6:7:8, 1:2:3:4::6:7:8, 1:2:3:4::8
|
|
"(" + ipv6segment + ":){1,4}(:" + ipv6segment + "){1,3}|" +
|
|
// 1::5:6:7:8, 1:2:3::5:6:7:8, 1:2:3::8
|
|
"(" + ipv6segment + ":){1,3}(:" + ipv6segment + "){1,4}|" +
|
|
// # 1::4:5:6:7:8, 1:2::4:5:6:7:8, 1:2::8
|
|
"(" + ipv6segment + ":){1,2}(:" + ipv6segment + "){1,5}|" +
|
|
// # 1::3:4:5:6:7:8, 1::3:4:5:6:7:8, 1::8
|
|
ipv6segment + ":((:" + ipv6segment + "){1,6})|" +
|
|
// ::2:3:4:5:6:7:8, ::2:3:4:5:6:7:8, ::8, ::
|
|
":((:" + ipv6segment + "){1,7}|:)|" +
|
|
// fe80::7:8%eth0, fe80::7:8%1 (link-local IPv6 addresses with
|
|
// zone index)
|
|
"fe80:(:" + ipv6segment + "){0,4}%[0-9a-zA-Z]{1,}|" +
|
|
// ::255.255.255.255, ::ffff:255.255.255.255,
|
|
// ::ffff:0:255.255.255.255 (IPv4-mapped IPv6 addresses and
|
|
// IPv4-translated addresses)
|
|
"::(ffff(:0{1,4}){0,1}:){0,1}" + ipv4address + "|" +
|
|
// 2001:db8:3:4::192.0.2.33, 64:ff9b::192.0.2.33 (IPv4-Embedded
|
|
// IPv6 Address)
|
|
"(" + ipv6segment + ":){1,4}:" + ipv4address + ")";
|
|
|
|
private final String ipv6addressWithPort = "\\[" + ipv6address + "\\]"
|
|
+ portsegment + "?";
|
|
|
|
/**
|
|
* Parses ipv4 and ipv6 addresses. Emits each described IP address as a
|
|
* hexadecimal integer representing the address, the address space, and the
|
|
* port number specified, if any.
|
|
*
|
|
* @param address the address to analyze
|
|
*/
|
|
public void parse(String address) {
|
|
|
|
// Used for storing values to be printed
|
|
String space = "";// ipv4, ipv6, or unknown
|
|
String hex = "";// hex value of the address
|
|
String port = "";// the port or unknown
|
|
|
|
// Try to match the pattern with one of the 2 types, with or without a
|
|
// port
|
|
if (Pattern.matches("^" + ipv4address + "$", address)) {
|
|
InetAddress a;
|
|
space = "IPv4";
|
|
try {
|
|
a = InetAddress.getByName(address);
|
|
hex = DatatypeConverter.printHexBinary(a.getAddress());
|
|
}
|
|
catch (UnknownHostException e) {
|
|
e.printStackTrace();
|
|
hex = "Invalid";
|
|
}
|
|
port = "Absent";
|
|
}
|
|
else if (Pattern.matches("^" + ipv4addressWithPort + "$", address)) {
|
|
String[] parts = address.split("\\.");
|
|
port = parts[3].split(":")[1];
|
|
parts[3] = parts[3].split(":")[0];
|
|
InetAddress a;
|
|
space = "IPv4";
|
|
try {
|
|
address = parts[0] + parts[1] + parts[2] + parts[3];
|
|
a = InetAddress.getByName(address);
|
|
hex = DatatypeConverter.printHexBinary(a.getAddress());
|
|
}
|
|
catch (UnknownHostException e) {
|
|
e.printStackTrace();
|
|
hex = "Invalid";
|
|
}
|
|
}
|
|
else if (Pattern.matches("^" + ipv6address + "$", address)) {
|
|
InetAddress a;
|
|
space = "IPv6";
|
|
try {
|
|
a = Inet6Address.getByName(address);
|
|
hex = DatatypeConverter.printHexBinary(a.getAddress());
|
|
}
|
|
catch (UnknownHostException e) {
|
|
e.printStackTrace();
|
|
hex = "Invalid";
|
|
}
|
|
port = "Absent";
|
|
}
|
|
else if (Pattern.matches("^" + ipv6addressWithPort + "$", address)) {
|
|
String[] parts = address.split(":");
|
|
InetAddress a;
|
|
space = "IPv6";
|
|
address =
|
|
address.replace("[", "").replace("]", "")
|
|
.replaceAll(portsegment + "$", "");
|
|
try {
|
|
a = Inet6Address.getByName(address);
|
|
hex = DatatypeConverter.printHexBinary(a.getAddress());
|
|
}
|
|
catch (UnknownHostException e) {
|
|
e.printStackTrace();
|
|
hex = "Invalid";
|
|
}
|
|
port = parts[parts.length - 1];
|
|
}
|
|
else {
|
|
// Not a valid address
|
|
hex = "Invalid";
|
|
space = "Invalid";
|
|
port = "Invalid";
|
|
}
|
|
|
|
// Output the findings to the console
|
|
System.out.println("Test case: '" + address + "'");
|
|
System.out.println("Space: " + space);
|
|
System.out.println("Address: " + hex);
|
|
System.out.println("Port: " + port);
|
|
System.out.println();
|
|
|
|
}
|
|
|
|
/**
|
|
* Tests the parser using various addresses.
|
|
*
|
|
* @param args arguments for the program
|
|
*/
|
|
public static void main(String[] args) {
|
|
IPParser parser = new IPParser();
|
|
|
|
// The "localhost" IPv4 address
|
|
parser.parse("127.0.0.1");
|
|
// The "localhost" IPv4 address, with a specified port (80)
|
|
parser.parse("127.0.0.1:80");
|
|
// The "localhost" IPv6 address
|
|
parser.parse("::1");
|
|
// The "localhost" IPv6 address, with a specified port (80)
|
|
parser.parse("[::1]:80");
|
|
// Rosetta Code's primary server's public IPv6 address
|
|
parser.parse("2605:2700:0:3::4713:93e3");
|
|
// Rosetta Code's primary server's public IPv6 address, with a specified
|
|
// port (80)
|
|
parser.parse("[2605:2700:0:3::4713:93e3]:80");
|
|
|
|
// ipv6 space
|
|
parser.parse("::ffff:192.168.173.22");
|
|
// ipv6 space with port
|
|
parser.parse("[::ffff:192.168.173.22]:80");
|
|
// trailing compression
|
|
parser.parse("1::");
|
|
// trailing compression with port
|
|
parser.parse("[1::]:80");
|
|
// 'any' address compression
|
|
parser.parse("::");
|
|
// 'any' address compression with port
|
|
parser.parse("[::]:80");
|
|
}
|
|
}
|