すべての学生がアカウントを持っている大学のLinuxサーバーにアクセスできます。を入力してフルネームを検索し、を入力してfinger username
部門を検索できますid username
。すべての学生のユーザー名は連続しています。たとえば、
e205846
e205847
e205848
e205849
e205850
...
一種のスクリプトを作成し、すべての情報を一種のデータベースで検索できますか?それともすでに利用可能なツールがありますか?また、参考までに、Linuxにはすべてのユーザー名を含むwwwフォルダがあります。
これは例です
e147290@beluga:~$ finger e204158
Login: e204158 Name: april oneil
Directory: /home705/e204158 Shell: /bin/bash
Never logged in.
No mail.
No Plan.
e147290@beluga:~$ id e204159
uid=53653(e204158) gid=5621(ce_bs) groups=5621(ce_bs)
答え1
Linuxシステムには、標準機能(ここで使用されている機能)をgetent
利用するプログラムがあります。検索したいのはデータベースの内容です(もっと理解してください)。get*ent(2)
getpwent()
passwd
man nsswitch.conf
$ getent passwd
root:x:0:0:root:/root:/bin/bash
sashroot:x:0:0:root:/root:/bin/sash
daemon:x:1:1:daemon:/usr/sbin:/bin/sh
bin:x:2:2:bin:/bin:/bin/sh
sys:x:3:3:sys:/dev:/bin/sh
sync:x:65004:65534:sync:/bin:/bin/sync
games:x:65005:65060:games:/usr/games:/bin/sh
man:x:65006:65012:man:/var/cache/man:/bin/sh
lp:x:65007:7:lp:/var/spool/lpd:/bin/sh
mail:x:65008:65008:mail:/var/mail:/bin/sh
...
標準システムアカウントがロールオーバーすると、より興味深くなります。
Linuxシステムでない場合は、この機能を使用してユーザーデータベースダンプを直接作成することで簡単にこれを達成できますgetpwent(2)
。 C言語の例は次のとおりです。
#include <sys/types.h>
#include <pwd.h>
#include <stdio.h>
int
main(void)
{
struct passwd *pwd;
while ((pwd = getpwent()) != NULL) {
printf("user = `%s', uid = %d, gid = %d, name = `%s'\n",
pwd->pw_name, pwd->pw_uid, pwd->pw_gid, pwd->pw_gecos);
}
return 0;
}
ファイルに入れて、test.c
次のようにビルドします。
$ gcc -o test test.c
その後、電話をかけて何が得られるかを確認できます./test
。
$ ./test
user = `root', uid = 0, gid = 0, name = `root'
user = `sashroot', uid = 0, gid = 0, name = `root'
user = `daemon', uid = 1, gid = 1, name = `daemon'
user = `bin', uid = 2, gid = 2, name = `bin'
user = `sys', uid = 3, gid = 3, name = `sys'
user = `sync', uid = 65004, gid = 65534, name = `sync'
user = `games', uid = 65005, gid = 65060, name = `games'
user = `man', uid = 65006, gid = 65012, name = `man'
user = `lp', uid = 65007, gid = 7, name = `lp'
user = `mail', uid = 65008, gid = 65008, name = `mail'
...
上記のように、後でより興味深くなります。
このアプローチを使用するgetwent()
最大の利点は、/etc/passwd
ログインシステムがどのように構成されているか(LDAP、NIS、...?)仮定する必要がなく、システムが実際の構成に基づいて情報を取得できるようにすることです。
編集する:passwd
データベースのストレージバックエンド(LDAPなど)がデータベース列挙(so getpwent()
)を許可せず、キーが明示的に要求されたデータセット(ログインまたはUID、したがって、またはgetpwuid()
有効である可能性がある)のみを提供する場合getpwnam()
。この場合(そしてターゲットユーザー名の名前は非常にきれいに指定されているので)、@masegaloehのスクリプトを修正して手動で「列挙する」ことができます。
#!/bin/bash
for i in {000000..999999}; do
getent passwd "e${i}" 2> /dev/null
done
getent
システムAPI(getpwnam()
、、)なしでgetpwuid()
これを行う方法は、読者に練習課題として残されます。
答え2
getentは、実際にはUnix / Linuxシステムの名前スイッチを介してすべてのデータベースにアクセスする正しい方法です。したがって、アカウントが/etc/passwdにローカルに保存されているのか、LDAP、AD、NISにあるのか、nsswitchで設定したのかを確認する必要があります。すべてのサービスをconfに設定します。
だから私はそうします:
getent passwd | awk -F: '$1 ~"^e[0-9]*$" {print $5}'
これにより、「e」で始まり、数字だけが続くすべてのアカウントの完全なユーザー名が印刷されます。
Windowsシステムでこれを行う必要がある場合は、cygwinをインストールして上記のようにします。 :) 実際、Windows/cygwinでgetentをテストしたところ、うまくいきます(やや驚くべきことです)。
答え3
このbashスクリプトはループを実行します。
#!/bin/bash
for i in {205846..205850}; do
finger e${i}
id e${i}
done
205846 および 205850 を変更して、異なる範囲を取得できます。
答え4
/etc/passwdからすべてのユーザー情報を取得できます。