Counting an array based on a condition: step-by-step guide
P粉133321839
P粉133321839 2023-09-10 13:15:11
0
2
615

I have a list of hosts with interfaces. The code must count the duplicate interfaces for each host. Finally, the code must also display the duplicate interface X times for each host.

I'm asking this question because I want to send alerts that host X has failed with one or more failed interfaces.

$data = array(
    array("192.168.0.1","eth1"),
    array("192.168.0.2","eth2"),
    array("192.168.0.3","eth3"),
    array("192.168.0.1","eth1"),
    array("192.168.0.4","eth1"),
    array("192.168.0.2","eth5")
);

I've followed other examples here, but most are simple arrays, or if it's a multidimensional example, the examples are not similar.

I already tried...

<?php
$data = array(
    array("192.168.0.1","eth1"),
    array("192.168.0.2","eth2"),
    array("192.168.0.3","eth3"),
    array("192.168.0.1","eth1"),
    array("192.168.0.4","eth1"),
    array("192.168.0.2","eth5")
);


$counter_data = count($data);

$duplicated_host = array_filter(array_count_values(array_column($data, 0)), function($v) { return $v > 1; });
print_r($duplicated_host);
print ("<br>");

$duplicated_host_keys = (array_keys($duplicated_host));

for ($row_num = 0; $row_num < $counter_data; $row_num++)
{
    $host = $data[$row_num][0];
    $interface = $data[$row_num][1];
    if (in_array($host,$duplicated_host_keys))
    {
        print($host . " " . $interface . "<br>");
    }
    
}

The code above is wrong, it works somewhat but it's not what I expected... Is there an easy way to do this?

The final output should look like this:

Host 192.168.0.1 has eth1 repeated 2 times. --> For current data only
Host 192.168.0.1 has eth9 repeated 5 times.
Host 192.168.0.4 has eth1 repeated 9 times.

P粉133321839
P粉133321839

reply all(2)
P粉124070451

This might be what you are looking for:

<?php
$input = array(
    array("192.168.0.1","eth1"),
    array("192.168.0.2","eth2"),
    array("192.168.0.3","eth3"),
    array("192.168.0.1","eth1"),
    array("192.168.0.4","eth1"),
    array("192.168.0.2","eth5"),
);
$output = [];
array_walk($input, function($entry) use (&$output) {
    [$host, $interface] = $entry;
    if (isset($host, $output) && isset($interface, $output[$host])) {
        $output[$host][$interface]++;
    } else {
        $output[$host][$interface] = 1;
    }
});
print_r($output);

The output is:

Array
(
    [192.168.0.1] => Array
        (
            [eth1] => 2
        )
    [192.168.0.2] => Array
        (
            [eth2] => 1
            [eth5] => 1
        )
    [192.168.0.3] => Array
        (
            [eth3] => 1
        )
    [192.168.0.4] => Array
        (
            [eth1] => 1
        )
)
P粉706038741

You need to group two groups, first the host and then the interface.

You can then loop over this grouped array to display/send the output:

<?php

$data = array(
    array("192.168.0.1","eth1"),
    array("192.168.0.2","eth2"),
    array("192.168.0.3","eth3"),
    array("192.168.0.1","eth1"),
    array("192.168.0.4","eth1"),
    array("192.168.0.2","eth5")
);


$result = [];

foreach ($data as $arr) {
    [ $host, $nic ] = $arr;
    if (!isset($result[$host])) {
        $result[$host] = [];
    }
    if (!isset($result[$host][$nic])) {
        $result[$host][$nic] = 0;
    }

    $result[$host][$nic]++;
}


foreach ($result as $host => $nics) {
    foreach ($nics as $nic => $count) {
        echo "${host} has his '${nic}' interface fail ${count} time(s)" . PHP_EOL;
    }
}
192.168.0.1 has his 'eth1' interface fail 2 time(s)
192.168.0.2 has his 'eth2' interface fail 1 time(s)
192.168.0.2 has his 'eth5' interface fail 1 time(s)
192.168.0.3 has his 'eth3' interface fail 1 time(s)
192.168.0.4 has his 'eth1' interface fail 1 time(s)

Try it online!


NIC -->"Network Interface Card"

Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template