grep을 사용하여 /etc/passwd에 나열된 모든 사용자 홈 디렉터리를 찾아야 합니다.

grep을 사용하여 /etc/passwd에 나열된 모든 사용자 홈 디렉터리를 찾아야 합니다.

/etc/passwd에서 모든 사용자 목록을 찾기 위해 grep이나 awk를 사용해야 하는 이 사이트의 다른 문제와 비슷한 문제가 있었습니다. 이것은 나에게 효과적이지만 홈 디렉토리를 찾아 나열하기 위해 번역하려고 합니다. 나는 이미 한 줄로 할 수 없다는 것을 알고 있으므로 파이프를 사용할 것이라는 것을 알고 있습니다. 온라인으로 조사를 해봤지만 문제가 무엇인지 알 수 없습니다. grep을 사용하여 다음과 같이 하면:

   grep -oE '^[/*/]$' /etc/passwd 

...오류가 발생하거나 내가 원하는 것이 아닌 /bin/bash 파일이 표시될 수도 있습니다. 나열된 사용자 이름과 해당 홈 디렉터리를 파악하면 됩니다! 또한 일부 홈 디렉토리에는 /(슬래시)가 두 개 이상 있기 때문에 *가 다른 슬래시를 문자로 표시할지 확실하지 않습니다.

답변1

당신은 그것을 사용할 수 있습니다cut특정 구분 기호의 열을 사용하여 파일을 분할합니다.

cut -d: -f6 /etc/passwd

또는 -f1,6열 1과 6(필드)의 경우.

답변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

관련 정보