/etc/passwd에서 모든 사용자 목록을 찾기 위해 grep이나 awk를 사용해야 하는 이 사이트의 다른 문제와 비슷한 문제가 있었습니다. 이것은 나에게 효과적이지만 홈 디렉토리를 찾아 나열하기 위해 번역하려고 합니다. 나는 이미 한 줄로 할 수 없다는 것을 알고 있으므로 파이프를 사용할 것이라는 것을 알고 있습니다. 온라인으로 조사를 해봤지만 문제가 무엇인지 알 수 없습니다. grep을 사용하여 다음과 같이 하면:
grep -oE '^[/*/]$' /etc/passwd
...오류가 발생하거나 내가 원하는 것이 아닌 /bin/bash 파일이 표시될 수도 있습니다. 나열된 사용자 이름과 해당 홈 디렉터리를 파악하면 됩니다! 또한 일부 홈 디렉토리에는 /(슬래시)가 두 개 이상 있기 때문에 *가 다른 슬래시를 문자로 표시할지 확실하지 않습니다.
답변1
답변2
Grep은 실제로 이런 방식으로 데이터를 구문 분석하는 도구가 아닙니다. grep은 패턴 일치에 더 적합하며 텍스트 처리를 수행하려고 합니다. awk를 사용하고 싶을 것입니다.
awk -F":" '$7 == "/bin/false" {print "User: "$1 "Home Dir: "$6}' /etc/passwd
awk
- 주문하다-F":"
– 데이터 구분 기호를 다음으로 설정합니다.:
$7 == "/bin/false"
– 7번째 데이터 열이 맞는지 확인/bin/false
{print "User: "$1 "Home Dir: "$6}'
– 첫 번째와 여섯 번째 열을 지정된 형식으로 인쇄하는 것을 의미합니다./etc/passwd
– 우리가 작업 중인 파일입니다.
답변3
다른 사람들이 지적했듯이 grep
이것은 최고의 도구는 아닙니다. 당신이 그것을 사용하기를 고집하고 (줄의 일치하는 부분만 인쇄) 및 (Perl 호환 정규식 사용)을 grep
지원 한다면 다음과 같이 할 수 있습니다:-o
-P
$ grep -oP '^[^:]+|.*:\K[^:]+(?=:[^:]+)' /etc/password
terdon
/home/terdon
bob
/home/bob
이는 시스템 사용자를 포함한 모든 사용자에 대해 인쇄됩니다. 예시로 4줄만 보여드리겠습니다.
그러면 모든 사용자의 사용자 이름과 홈 디렉토리가 별도의 줄에 인쇄됩니다. 그런 다음 각 쌍의 전선을 연결하여 함께 모아야 합니다.
$ grep -oP '^[^:]+|.*:\K[^:]+(?=:[^:]+)' /etc/passwd | perl -pe 's/\n/ : / if $.%2'
root : /root
bin : /bin
daemon : /
mail : /var/spool/mail
ftp : /srv/ftp
http : /srv/http
uuidd : /
dbus : /
nobody : /
systemd-journal-gateway : /
systemd-timesync : /
systemd-network : /
systemd-bus-proxy : /
systemd-resolve : /
systemd-journal-upload : /
systemd-coredump : /
systemd-journal-remote : /
terdon : /home/terdon
avahi : /
polkitd : /
colord : /var/lib/colord
rtkit : /proc
gdm : /var/lib/gdm
git : /
bob : /home/bob
설명하다
정규식은 두 부분으로 구성되어 있습니다. 즉, ^[^:]+
OR(이것이 바로 |
) 를 찾는 것입니다 .*:\K[^:]+(?=:[^:]+)
. 첫 번째는 :
줄의 시작 부분에서 하나 이상의 문자가 아닌 항목을 찾습니다. 사용자 이름과 일치합니다. 두 번째 부분은 :
( ) 까지 가능한 한 많은 문자를 찾은 다음 .*:
해당 문자를 삭제하여(이렇게 \K
합니다) 인쇄되지 않습니다. 그런 다음 문자열 non- :
와 그 뒤에 :
and non- 가 일치합니다 :
. 이 (?=foo)
구조를긍정적 인 전망문자를 일치시키는 방법입니다뒤쪽에이러한 문자를 포함하지 않는 패턴은 자체적으로 일치합니다.
현재 줄 번호( )가 2로 나눌 수 있는 경우 이 perl
명령은 줄 바꿈을 공백으로 바꿉니다. 그래서 다른 모든 행.:
$.
답변4
나는 당신이 하나의 바이너리만을 사용하고, 파이프를 피하고, 다른 답변과 동일한 결과를 달성하면서 "절단"함으로써 그것을 할 수 있다고 믿습니다. 그러나 더 우아한 방법으로 :) 다음과 같이:
$ cut -d : -f 1,6 /etc/passwd
root:/root
daemon:/usr/sbin
bin:/bin
sys:/dev
sync:/bin
games:/usr/games
man:/var/cache/man
lp:/var/spool/lpd
mail:/var/mail
news:/var/spool/news
....
더 나은 형식 + 알파벳순 출력을 원한다면 여기에 있지만 더 많은 바이너리를 사용해야 한다는 단점이 있습니다.
$ cut -d : -f 1,6 /etc/passwd | sort | column
avahi-autoipd:/var/lib/avahi-autoipd man:/var/cache/man
avahi:/var/run/avahi-daemon messagebus:/var/run/dbus
backup:/var/backups news:/var/spool/news
bin:/bin nobody:/nonexistent
clickpkg:/nonexistent ntp:/home/ntp
colord:/var/lib/colord proxy:/bin
daemon:/usr/sbin pulse:/var/run/pulse
dnsmasq:/var/lib/misc root:/root
games:/usr/games rtkit:/proc
gnats:/var/lib/gnats saned:/home/saned
hplip:/var/run/hplip speech-dispatcher:/var/run/speech-dispatcher
irc:/var/run/ircd sync:/bin
ivanleon:/home/ivanleon sys:/dev
kernoops:/ syslog:/home/syslog
libuuid:/var/lib/libuuid usbmux:/home/usbmux
lightdm:/var/lib/lightdm usermetrics:/var/lib/usermetrics
list:/var/list uucp:/var/spool/uucp
lp:/var/spool/lpd whoopsie:/nonexistent
lxc-dnsmasq:/var/lib/lxc www-data:/var/www
mail:/var/mail