oshi.software.os.linux.LinuxInternetProtocolStats Maven / Gradle / Ivy
The newest version!
/*
* Copyright 2020-2024 The OSHI Project Contributors
* SPDX-License-Identifier: MIT
*/
package oshi.software.os.linux;
import static oshi.software.os.InternetProtocolStats.TcpState.CLOSED;
import static oshi.software.os.InternetProtocolStats.TcpState.CLOSE_WAIT;
import static oshi.software.os.InternetProtocolStats.TcpState.CLOSING;
import static oshi.software.os.InternetProtocolStats.TcpState.ESTABLISHED;
import static oshi.software.os.InternetProtocolStats.TcpState.FIN_WAIT_1;
import static oshi.software.os.InternetProtocolStats.TcpState.FIN_WAIT_2;
import static oshi.software.os.InternetProtocolStats.TcpState.LAST_ACK;
import static oshi.software.os.InternetProtocolStats.TcpState.LISTEN;
import static oshi.software.os.InternetProtocolStats.TcpState.SYN_RECV;
import static oshi.software.os.InternetProtocolStats.TcpState.SYN_SENT;
import static oshi.software.os.InternetProtocolStats.TcpState.TIME_WAIT;
import static oshi.software.os.InternetProtocolStats.TcpState.UNKNOWN;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import oshi.annotation.concurrent.ThreadSafe;
import oshi.driver.linux.proc.ProcessStat;
import oshi.driver.unix.NetStat;
import oshi.software.common.AbstractInternetProtocolStats;
import oshi.util.FileUtil;
import oshi.util.ParseUtil;
import oshi.util.platform.linux.ProcPath;
import oshi.util.tuples.Pair;
/**
* Internet Protocol Stats implementation
*/
@ThreadSafe
public class LinuxInternetProtocolStats extends AbstractInternetProtocolStats {
@Override
public TcpStats getTCPv4Stats() {
return NetStat.queryTcpStats("netstat -st4");
}
@Override
public UdpStats getUDPv4Stats() {
return NetStat.queryUdpStats("netstat -su4");
}
@Override
public UdpStats getUDPv6Stats() {
return NetStat.queryUdpStats("netstat -su6");
}
@Override
public List getConnections() {
List conns = new ArrayList<>();
Map pidMap = ProcessStat.querySocketToPidMap();
conns.addAll(queryConnections("tcp", 4, pidMap));
conns.addAll(queryConnections("tcp", 6, pidMap));
conns.addAll(queryConnections("udp", 4, pidMap));
conns.addAll(queryConnections("udp", 6, pidMap));
return conns;
}
private static List queryConnections(String protocol, int ipver, Map pidMap) {
List conns = new ArrayList<>();
for (String s : FileUtil.readFile(ProcPath.NET + "/" + protocol + (ipver == 6 ? "6" : ""))) {
if (s.indexOf(':') >= 0) {
String[] split = ParseUtil.whitespaces.split(s.trim());
if (split.length > 9) {
Pair lAddr = parseIpAddr(split[1]);
Pair fAddr = parseIpAddr(split[2]);
TcpState state = stateLookup(ParseUtil.hexStringToInt(split[3], 0));
Pair txQrxQ = parseHexColonHex(split[4]);
long inode = ParseUtil.parseLongOrDefault(split[9], 0);
conns.add(new IPConnection(protocol + ipver, lAddr.getA(), lAddr.getB(), fAddr.getA(), fAddr.getB(),
state, txQrxQ.getA(), txQrxQ.getB(), pidMap.getOrDefault(inode, -1)));
}
}
}
return conns;
}
private static Pair parseIpAddr(String s) {
int colon = s.indexOf(':');
if (colon > 0 && colon < s.length()) {
byte[] first = ParseUtil.hexStringToByteArray(s.substring(0, colon));
// Bytes are in __be32 endianness. we must invert each set of 4 bytes
for (int i = 0; i + 3 < first.length; i += 4) {
byte tmp = first[i];
first[i] = first[i + 3];
first[i + 3] = tmp;
tmp = first[i + 1];
first[i + 1] = first[i + 2];
first[i + 2] = tmp;
}
int second = ParseUtil.hexStringToInt(s.substring(colon + 1), 0);
return new Pair<>(first, second);
}
return new Pair<>(new byte[0], 0);
}
private static Pair parseHexColonHex(String s) {
int colon = s.indexOf(':');
if (colon > 0 && colon < s.length()) {
int first = ParseUtil.hexStringToInt(s.substring(0, colon), 0);
int second = ParseUtil.hexStringToInt(s.substring(colon + 1), 0);
return new Pair<>(first, second);
}
return new Pair<>(0, 0);
}
private static TcpState stateLookup(int state) {
switch (state) {
case 0x01:
return ESTABLISHED;
case 0x02:
return SYN_SENT;
case 0x03:
return SYN_RECV;
case 0x04:
return FIN_WAIT_1;
case 0x05:
return FIN_WAIT_2;
case 0x06:
return TIME_WAIT;
case 0x07:
return CLOSED;
case 0x08:
return CLOSE_WAIT;
case 0x09:
return LAST_ACK;
case 0x0A:
return LISTEN;
case 0x0B:
return CLOSING;
case 0x00:
default:
return UNKNOWN;
}
}
}