2021-05-08 10:46:14 -04:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Brandon Pruitt <brapru@pm.me>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
#include <AK/DeprecatedString.h>
|
2021-07-12 21:23:36 -04:00
|
|
|
#include <AK/HashMap.h>
|
2022-04-17 10:08:00 -04:00
|
|
|
#include <AK/IPv4Address.h>
|
2021-05-08 10:46:14 -04:00
|
|
|
#include <AK/JsonArray.h>
|
|
|
|
#include <AK/JsonObject.h>
|
|
|
|
#include <AK/QuickSort.h>
|
|
|
|
#include <LibCore/ArgsParser.h>
|
2023-02-09 03:02:46 +01:00
|
|
|
#include <LibCore/File.h>
|
2021-07-12 21:23:36 -04:00
|
|
|
#include <LibCore/ProcessStatisticsReader.h>
|
2021-11-29 21:26:35 +01:00
|
|
|
#include <LibCore/System.h>
|
|
|
|
#include <LibMain/Main.h>
|
2022-04-03 09:13:28 +00:00
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <netdb.h>
|
2021-05-08 10:46:14 -04:00
|
|
|
#include <unistd.h>
|
|
|
|
|
2022-04-16 16:18:13 -04:00
|
|
|
constexpr int max_formatted_address_length = 21;
|
|
|
|
|
2021-11-29 21:26:35 +01:00
|
|
|
ErrorOr<int> serenity_main(Main::Arguments arguments)
|
2021-05-08 10:46:14 -04:00
|
|
|
{
|
2022-04-17 10:08:00 -04:00
|
|
|
TRY(Core::System::pledge("stdio rpath unix"));
|
2021-05-08 10:46:14 -04:00
|
|
|
|
|
|
|
bool flag_all = false;
|
|
|
|
bool flag_list = false;
|
|
|
|
bool flag_tcp = false;
|
|
|
|
bool flag_udp = false;
|
2022-04-03 09:13:28 +00:00
|
|
|
bool flag_numeric = false;
|
2021-07-12 21:23:36 -04:00
|
|
|
bool flag_program = false;
|
2022-04-16 16:18:13 -04:00
|
|
|
bool flag_wide = false;
|
2021-05-08 10:46:14 -04:00
|
|
|
|
|
|
|
Core::ArgsParser args_parser;
|
|
|
|
args_parser.set_general_help("Display network connections");
|
|
|
|
args_parser.add_option(flag_all, "Display both listening and non-listening sockets", "all", 'a');
|
|
|
|
args_parser.add_option(flag_list, "Display only listening sockets", "list", 'l');
|
|
|
|
args_parser.add_option(flag_tcp, "Display only TCP network connections", "tcp", 't');
|
|
|
|
args_parser.add_option(flag_udp, "Display only UDP network connections", "udp", 'u');
|
2022-04-03 09:13:28 +00:00
|
|
|
args_parser.add_option(flag_numeric, "Display numerical addresses", "numeric", 'n');
|
2021-07-12 21:23:36 -04:00
|
|
|
args_parser.add_option(flag_program, "Show the PID and name of the program to which each socket belongs", "program", 'p');
|
2022-04-16 16:18:13 -04:00
|
|
|
args_parser.add_option(flag_wide, "Do not truncate IP addresses by printing out the whole symbolic host", "wide", 'W');
|
2021-11-29 21:26:35 +01:00
|
|
|
args_parser.parse(arguments);
|
2021-05-08 10:46:14 -04:00
|
|
|
|
2022-10-14 21:56:19 +03:00
|
|
|
TRY(Core::System::unveil("/sys/kernel/net", "r"));
|
|
|
|
TRY(Core::System::unveil("/sys/kernel/processes", "r"));
|
2022-05-26 19:28:45 +02:00
|
|
|
TRY(Core::System::unveil("/etc/passwd", "r"));
|
|
|
|
TRY(Core::System::unveil("/etc/services", "r"));
|
2023-02-11 17:42:43 +02:00
|
|
|
if (!flag_numeric)
|
|
|
|
TRY(Core::System::unveil("/tmp/portal/lookup", "rw"));
|
|
|
|
|
2022-05-26 19:28:45 +02:00
|
|
|
TRY(Core::System::unveil(nullptr, nullptr));
|
|
|
|
|
2021-05-08 10:46:14 -04:00
|
|
|
bool has_protocol_flag = (flag_tcp || flag_udp);
|
|
|
|
|
2021-07-12 21:23:36 -04:00
|
|
|
uid_t current_uid = getuid();
|
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
HashMap<pid_t, DeprecatedString> programs;
|
2021-07-12 21:23:36 -04:00
|
|
|
|
|
|
|
if (flag_program) {
|
2022-12-08 14:50:31 +01:00
|
|
|
auto processes = TRY(Core::ProcessStatisticsReader::get_all());
|
2021-07-12 21:23:36 -04:00
|
|
|
|
2022-12-08 14:50:31 +01:00
|
|
|
for (auto& proc : processes.processes) {
|
2021-07-12 21:23:36 -04:00
|
|
|
programs.set(proc.pid, proc.name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-08 10:46:14 -04:00
|
|
|
enum class Alignment {
|
|
|
|
Left,
|
|
|
|
Right
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Column {
|
2022-12-04 18:02:33 +00:00
|
|
|
DeprecatedString title;
|
2021-05-08 10:46:14 -04:00
|
|
|
Alignment alignment { Alignment::Left };
|
|
|
|
int width { 0 };
|
2022-12-04 18:02:33 +00:00
|
|
|
DeprecatedString buffer;
|
2021-05-08 10:46:14 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
Vector<Column> columns;
|
|
|
|
|
|
|
|
int protocol_column = -1;
|
|
|
|
int bytes_in_column = -1;
|
|
|
|
int bytes_out_column = -1;
|
|
|
|
int local_address_column = -1;
|
|
|
|
int peer_address_column = -1;
|
|
|
|
int state_column = -1;
|
2021-07-12 21:23:36 -04:00
|
|
|
int program_column = -1;
|
2021-05-08 10:46:14 -04:00
|
|
|
|
|
|
|
auto add_column = [&](auto title, auto alignment, auto width) {
|
|
|
|
columns.append({ title, alignment, width, {} });
|
|
|
|
return columns.size() - 1;
|
|
|
|
};
|
|
|
|
|
|
|
|
protocol_column = add_column("Proto", Alignment::Left, 5);
|
|
|
|
bytes_in_column = add_column("Bytes-In", Alignment::Right, 9);
|
|
|
|
bytes_out_column = add_column("Bytes-Out", Alignment::Right, 9);
|
2022-04-16 16:18:13 -04:00
|
|
|
local_address_column = add_column("Local Address", Alignment::Left, 22);
|
|
|
|
peer_address_column = add_column("Peer Address", Alignment::Left, 22);
|
2021-07-12 21:23:36 -04:00
|
|
|
state_column = add_column("State", Alignment::Left, 11);
|
|
|
|
program_column = flag_program ? add_column("PID/Program", Alignment::Left, 11) : -1;
|
2021-05-08 10:46:14 -04:00
|
|
|
|
|
|
|
auto print_column = [](auto& column, auto& string) {
|
|
|
|
if (!column.width) {
|
|
|
|
out("{}", string);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (column.alignment == Alignment::Right) {
|
|
|
|
out("{:>{1}} "sv, string, column.width);
|
|
|
|
} else {
|
|
|
|
out("{:<{1}} "sv, string, column.width);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
auto get_formatted_address = [&](DeprecatedString const& address, DeprecatedString const& port) {
|
2022-04-16 16:18:13 -04:00
|
|
|
if (flag_wide)
|
2022-12-04 18:02:33 +00:00
|
|
|
return DeprecatedString::formatted("{}:{}", address, port);
|
2022-04-16 16:18:13 -04:00
|
|
|
|
|
|
|
if ((address.length() + port.length()) <= max_formatted_address_length)
|
2022-12-04 18:02:33 +00:00
|
|
|
return DeprecatedString::formatted("{}:{}", address, port);
|
2022-04-16 16:18:13 -04:00
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
return DeprecatedString::formatted("{}:{}", address.substring_view(0, max_formatted_address_length - port.length()), port);
|
2022-04-16 16:18:13 -04:00
|
|
|
};
|
|
|
|
|
2021-07-12 21:23:36 -04:00
|
|
|
auto get_formatted_program = [&](pid_t pid) {
|
|
|
|
if (pid == -1)
|
2022-12-04 18:02:33 +00:00
|
|
|
return DeprecatedString("-");
|
2021-07-12 21:23:36 -04:00
|
|
|
|
|
|
|
auto program = programs.get(pid);
|
2022-12-04 18:02:33 +00:00
|
|
|
return DeprecatedString::formatted("{}/{}", pid, program.value());
|
2021-07-12 21:23:36 -04:00
|
|
|
};
|
|
|
|
|
2021-05-08 10:46:14 -04:00
|
|
|
if (!has_protocol_flag || flag_tcp || flag_udp) {
|
2021-07-12 21:23:36 -04:00
|
|
|
if (flag_program && current_uid != 0) {
|
|
|
|
outln("(Some processes could not be identified, non-owned process info will not be shown)");
|
|
|
|
}
|
|
|
|
|
2021-05-08 10:46:14 -04:00
|
|
|
out("Active Internet connections ");
|
|
|
|
|
|
|
|
if (flag_all) {
|
|
|
|
outln("(servers and established)");
|
|
|
|
} else {
|
|
|
|
if (flag_list)
|
|
|
|
outln("(only servers)");
|
|
|
|
else
|
|
|
|
outln("(without servers)");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& column : columns)
|
|
|
|
print_column(column, column.title);
|
|
|
|
outln();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_protocol_flag || flag_tcp) {
|
2023-02-09 03:02:46 +01:00
|
|
|
auto file = TRY(Core::File::open("/sys/kernel/net/tcp"sv, Core::File::OpenMode::Read));
|
2022-12-11 17:49:00 +01:00
|
|
|
auto file_contents = TRY(file->read_until_eof());
|
2021-12-25 16:41:14 +01:00
|
|
|
auto json_or_error = JsonValue::from_string(file_contents);
|
|
|
|
if (json_or_error.is_error()) {
|
|
|
|
warnln("Error: {}", json_or_error.error());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
auto json = json_or_error.release_value();
|
2021-05-08 10:46:14 -04:00
|
|
|
|
2021-11-15 01:46:51 +01:00
|
|
|
Vector<JsonValue> sorted_regions = json.as_array().values();
|
2021-05-08 10:46:14 -04:00
|
|
|
quick_sort(sorted_regions, [](auto& a, auto& b) {
|
2022-12-22 14:28:33 +00:00
|
|
|
return a.as_object().get_u32("local_port"sv).value_or(0) < b.as_object().get_u32("local_port"sv).value_or(0);
|
2021-05-08 10:46:14 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
for (auto& value : sorted_regions) {
|
2021-05-31 17:59:02 +01:00
|
|
|
auto& if_object = value.as_object();
|
2021-05-08 10:46:14 -04:00
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto bytes_in = if_object.get_deprecated_string("bytes_in"sv).value_or({});
|
|
|
|
auto bytes_out = if_object.get_deprecated_string("bytes_out"sv).value_or({});
|
2022-04-17 10:08:00 -04:00
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto peer_address = if_object.get_deprecated_string("peer_address"sv).value_or({});
|
2022-04-17 10:08:00 -04:00
|
|
|
if (!flag_numeric) {
|
|
|
|
auto from_string = IPv4Address::from_string(peer_address);
|
|
|
|
auto addr = from_string.value().to_in_addr_t();
|
|
|
|
auto* hostent = gethostbyaddr(&addr, sizeof(in_addr), AF_INET);
|
|
|
|
if (hostent != nullptr) {
|
2022-07-11 19:53:29 +00:00
|
|
|
auto host_name = StringView { hostent->h_name, strlen(hostent->h_name) };
|
2022-04-17 10:08:00 -04:00
|
|
|
if (!host_name.is_empty())
|
|
|
|
peer_address = host_name;
|
|
|
|
}
|
|
|
|
}
|
2022-04-03 09:13:28 +00:00
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto peer_port = if_object.get_deprecated_string("peer_port"sv).value_or({});
|
2022-04-03 09:13:28 +00:00
|
|
|
if (!flag_numeric) {
|
2022-12-22 14:28:33 +00:00
|
|
|
auto service = getservbyport(htons(if_object.get_u32("peer_port"sv).value_or(0)), "tcp");
|
2022-04-03 09:13:28 +00:00
|
|
|
if (service != nullptr) {
|
2022-07-11 19:53:29 +00:00
|
|
|
auto s_name = StringView { service->s_name, strlen(service->s_name) };
|
2022-04-03 09:13:28 +00:00
|
|
|
if (!s_name.is_empty())
|
|
|
|
peer_port = s_name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto local_address = if_object.get_deprecated_string("local_address"sv).value_or({});
|
2022-04-17 10:08:00 -04:00
|
|
|
if (!flag_numeric) {
|
|
|
|
auto from_string = IPv4Address::from_string(local_address);
|
|
|
|
auto addr = from_string.value().to_in_addr_t();
|
|
|
|
auto* hostent = gethostbyaddr(&addr, sizeof(in_addr), AF_INET);
|
|
|
|
if (hostent != nullptr) {
|
2022-07-11 19:53:29 +00:00
|
|
|
auto host_name = StringView { hostent->h_name, strlen(hostent->h_name) };
|
2022-04-17 10:08:00 -04:00
|
|
|
if (!host_name.is_empty())
|
|
|
|
local_address = host_name;
|
|
|
|
}
|
|
|
|
}
|
2022-04-03 09:13:28 +00:00
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto local_port = if_object.get_deprecated_string("local_port"sv).value_or({});
|
2022-04-03 09:13:28 +00:00
|
|
|
if (!flag_numeric) {
|
2022-12-22 14:28:33 +00:00
|
|
|
auto service = getservbyport(htons(if_object.get_u32("local_port"sv).value_or(0)), "tcp");
|
2022-04-03 09:13:28 +00:00
|
|
|
if (service != nullptr) {
|
2022-07-11 19:53:29 +00:00
|
|
|
auto s_name = StringView { service->s_name, strlen(service->s_name) };
|
2022-04-03 09:13:28 +00:00
|
|
|
if (!s_name.is_empty())
|
|
|
|
local_port = s_name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto state = if_object.get_deprecated_string("state"sv).value_or({});
|
|
|
|
auto origin_pid = (if_object.has("origin_pid"sv)) ? if_object.get_u32("origin_pid"sv).value_or(0) : -1;
|
2021-05-08 10:46:14 -04:00
|
|
|
|
|
|
|
if (!flag_all && ((state == "Listen" && !flag_list) || (state != "Listen" && flag_list)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (protocol_column != -1)
|
|
|
|
columns[protocol_column].buffer = "tcp";
|
|
|
|
if (bytes_in_column != -1)
|
|
|
|
columns[bytes_in_column].buffer = bytes_in;
|
|
|
|
if (bytes_out_column != -1)
|
|
|
|
columns[bytes_out_column].buffer = bytes_out;
|
|
|
|
if (local_address_column != -1)
|
2022-04-16 16:18:13 -04:00
|
|
|
columns[local_address_column].buffer = get_formatted_address(local_address, local_port);
|
2021-05-08 10:46:14 -04:00
|
|
|
if (peer_address_column != -1)
|
2022-04-16 16:18:13 -04:00
|
|
|
columns[peer_address_column].buffer = get_formatted_address(peer_address, peer_port);
|
2021-05-08 10:46:14 -04:00
|
|
|
if (state_column != -1)
|
|
|
|
columns[state_column].buffer = state;
|
2021-07-12 21:23:36 -04:00
|
|
|
if (flag_program && program_column != -1)
|
|
|
|
columns[program_column].buffer = get_formatted_program(origin_pid);
|
2021-05-08 10:46:14 -04:00
|
|
|
|
|
|
|
for (auto& column : columns)
|
|
|
|
print_column(column, column.buffer);
|
|
|
|
outln();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_protocol_flag || flag_udp) {
|
2023-02-09 03:02:46 +01:00
|
|
|
auto file = TRY(Core::File::open("/sys/kernel/net/udp"sv, Core::File::OpenMode::Read));
|
2022-12-11 17:49:00 +01:00
|
|
|
auto file_contents = TRY(file->read_until_eof());
|
2021-11-29 21:26:35 +01:00
|
|
|
auto json = TRY(JsonValue::from_string(file_contents));
|
2021-05-08 10:46:14 -04:00
|
|
|
|
2021-11-15 01:46:51 +01:00
|
|
|
Vector<JsonValue> sorted_regions = json.as_array().values();
|
2021-05-08 10:46:14 -04:00
|
|
|
quick_sort(sorted_regions, [](auto& a, auto& b) {
|
2022-12-22 14:28:33 +00:00
|
|
|
return a.as_object().get_u32("local_port"sv).value_or(0) < b.as_object().get_u32("local_port"sv).value_or(0);
|
2021-05-08 10:46:14 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
for (auto& value : sorted_regions) {
|
2021-05-31 17:59:02 +01:00
|
|
|
auto& if_object = value.as_object();
|
2021-05-08 10:46:14 -04:00
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto local_address = if_object.get_deprecated_string("local_address"sv).value_or({});
|
2022-04-17 10:08:00 -04:00
|
|
|
if (!flag_numeric) {
|
|
|
|
auto from_string = IPv4Address::from_string(local_address);
|
|
|
|
auto addr = from_string.value().to_in_addr_t();
|
|
|
|
auto* hostent = gethostbyaddr(&addr, sizeof(in_addr), AF_INET);
|
|
|
|
if (hostent != nullptr) {
|
2022-07-11 19:53:29 +00:00
|
|
|
auto host_name = StringView { hostent->h_name, strlen(hostent->h_name) };
|
2022-04-17 10:08:00 -04:00
|
|
|
if (!host_name.is_empty())
|
|
|
|
local_address = host_name;
|
|
|
|
}
|
|
|
|
}
|
2022-04-03 09:13:28 +00:00
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto local_port = if_object.get_deprecated_string("local_port"sv).value_or({});
|
2022-04-03 09:13:28 +00:00
|
|
|
if (!flag_numeric) {
|
2022-12-22 14:28:33 +00:00
|
|
|
auto service = getservbyport(htons(if_object.get_u32("local_port"sv).value_or(0)), "udp");
|
2022-04-03 09:13:28 +00:00
|
|
|
if (service != nullptr) {
|
2022-07-11 19:53:29 +00:00
|
|
|
auto s_name = StringView { service->s_name, strlen(service->s_name) };
|
2022-04-03 09:13:28 +00:00
|
|
|
if (!s_name.is_empty())
|
|
|
|
local_port = s_name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto peer_address = if_object.get_deprecated_string("peer_address"sv).value_or({});
|
2022-04-17 10:08:00 -04:00
|
|
|
if (!flag_numeric) {
|
|
|
|
auto from_string = IPv4Address::from_string(peer_address);
|
|
|
|
auto addr = from_string.value().to_in_addr_t();
|
|
|
|
auto* hostent = gethostbyaddr(&addr, sizeof(in_addr), AF_INET);
|
|
|
|
if (hostent != nullptr) {
|
2022-07-11 19:53:29 +00:00
|
|
|
auto host_name = StringView { hostent->h_name, strlen(hostent->h_name) };
|
2022-04-17 10:08:00 -04:00
|
|
|
if (!host_name.is_empty())
|
|
|
|
peer_address = host_name;
|
|
|
|
}
|
|
|
|
}
|
2022-04-03 09:13:28 +00:00
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto peer_port = if_object.get_deprecated_string("peer_port"sv).value_or({});
|
2022-04-03 09:13:28 +00:00
|
|
|
if (!flag_numeric) {
|
2022-12-22 14:28:33 +00:00
|
|
|
auto service = getservbyport(htons(if_object.get_u32("peer_port"sv).value_or(0)), "udp");
|
2022-04-03 09:13:28 +00:00
|
|
|
if (service != nullptr) {
|
2022-07-11 19:53:29 +00:00
|
|
|
auto s_name = StringView { service->s_name, strlen(service->s_name) };
|
2022-04-03 09:13:28 +00:00
|
|
|
if (!s_name.is_empty())
|
|
|
|
peer_port = s_name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-22 14:28:33 +00:00
|
|
|
auto origin_pid = (if_object.has("origin_pid"sv)) ? if_object.get_u32("origin_pid"sv).value_or(0) : -1;
|
2021-05-08 10:46:14 -04:00
|
|
|
|
|
|
|
if (protocol_column != -1)
|
|
|
|
columns[protocol_column].buffer = "udp";
|
|
|
|
if (bytes_in_column != -1)
|
|
|
|
columns[bytes_in_column].buffer = "-";
|
|
|
|
if (bytes_out_column != -1)
|
|
|
|
columns[bytes_out_column].buffer = "-";
|
|
|
|
if (local_address_column != -1)
|
2022-04-16 16:18:13 -04:00
|
|
|
columns[local_address_column].buffer = get_formatted_address(local_address, local_port);
|
2021-05-08 10:46:14 -04:00
|
|
|
if (peer_address_column != -1)
|
2022-04-16 16:18:13 -04:00
|
|
|
columns[peer_address_column].buffer = get_formatted_address(peer_address, peer_port);
|
2021-05-08 10:46:14 -04:00
|
|
|
if (state_column != -1)
|
|
|
|
columns[state_column].buffer = "-";
|
2021-07-12 21:23:36 -04:00
|
|
|
if (flag_program && program_column != -1)
|
|
|
|
columns[program_column].buffer = get_formatted_program(origin_pid);
|
2021-05-08 10:46:14 -04:00
|
|
|
|
|
|
|
for (auto& column : columns)
|
|
|
|
print_column(column, column.buffer);
|
|
|
|
outln();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|