ファイルがあるとしましょう。
uid=1(a) groups=cat,dog,hen
uid=2(b) groups=cat,hen
uid=3(c) groups=hen
uid=4(d) groups=dog,hen,buffalo,horse
私が探しています:
cat a,b
dog a,d
hen a,b,c,d
buffalo d
horse d
リスト#1:ユーザーとセカンダリグループ(ユーザー別)
リスト#2:グループとユーザー(グループに属する)
答え1
awk -F "[= ,()]" '{
for(i=6;i<=NF;i++){
if(a[$i] != ""){
a[$i]=a[$i]","$3
}else{
a[$i]=$3
}
}
}
END{
for ( i in a){
print i,a[i]
}
}'
各グループメンバーで索引付けされた配列を作成し、uid
グループメンバーが行に見つかったら配列に追加します。最初はEND
配列インデックスと配列の値です。
答え2
#!/usr/bin/env perl
use strict;
use warnings;
use feature qw(say);
my %groups;
while ( my $line = readline *STDIN ) {
chomp $line;
if ( $line =~ m/\((?<user>[^)]+)\) groups=(?<groups>.+)/ ) {
for my $groupname ( split ',', $+{groups} ) {
push @{ $groups{$groupname} }, $+{user};
}
} else {
warn "NOTICE non-matching line $. - '$line'\n";
}
}
for my $groupname ( sort keys %groups ) {
say $groupname, ' ', join ',', @{ $groups{$groupname} };
}
このバージョンは配列参照(ユーザー)のハッシュ(グループ)を使用し、入力行と一致しない場合に警告します(データが破損していますか?どうすればわかりますか?)。
答え3
そしてperl
:
perl -lne '
if (/^uid=\d+\((.+)\) groups=(.*)/) {
push @{$u{$_}}, $1 for split /,/, $2;
}
END {
$" = ",";
print "$_ @{$u{$_}}" for keys %u;
}'