sudoでコマンドを実行し、sudo権限がタイムアウトしたときにのみテキスト行をエコーするスクリプトを作成しています。したがって、sudoでコマンドを実行するには、ユーザー(rootではない)がパスワードを再入力する必要があります。
これをどのように確認しますか? sudoで実行しても$(id -u)
現在のユーザーIDが返されるため、0に一致するかどうかを確認する方法はありません。
静かに確認する方法が必要です。
答え1
-n
以下の権限がまだあることを確認するには、このオプションを使用してくださいman sudo
。
-N、--非対話型
ユーザーに何らかのタイプの入力を要求しないでください。コマンドの実行にパスワードが必要な場合、sudoはエラーメッセージを表示して終了します。
例えば、
sudo -n true 2>/dev/null && echo Privileges active || echo Privileges inactive
sudo -n true
確認後、実際の使用の間に権限が期限切れになることがあります。自分で試してsudo -n command...
失敗した場合は、メッセージを表示してsudo
対話的にやり直してください。
編集:以下のruakhのコメントも参照してください。
答え2
ランニング:
sudo -nv
sudo権限が期限切れになると、終了コード1で終了し、次の内容が出力されます。
sudo: a password is required
有効なキャッシュされた資格情報がある場合、このコマンドは成功し、何も出力しません。
すべてをまとめると、次のスクリプトがあります。何も言わずに有効なキャッシュされた資格情報があることを確認してください。
if sudo -nv 2>/dev/null; then
echo "no sudo password required"
else
echo "sudo password expired"
fi
他の回答/コメントで述べたように、-v
sudoのオプション(「認証」)はキャッシュされた資格情報を生成するための認証プロンプトがある場合はキャッシュされた資格情報を自動的に更新し、オプション-n
(「非対話型」)はsudoです。同じインタラクティブなプロンプトが生成されるのを防ぎます。認証プロンプトで。
答え3
sudo -nv
うまく動作しますが、sudoエラーとpam認証メッセージでシステムログを汚染します。 sudo権限のbashプロンプトを確認する必要があるため、非常に頻繁に実行され、私のログにはほとんどこれらのノイズのみが含まれています。
sudoタイムスタンプファイルを直接解析することができます。これを行うために小さなCユーティリティを作成しました。
/* compile and set permissions: */
/* $ gcc checksudo.c -o checksudo -std=gnu99 -O2 */
/* $ chown root:root checksudo */
/* $ chmod +s checksudo */
#define USERNAME "replace-with-your-username"
#define TIMEOUT 5
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <time.h>
void timespec_diff(struct timespec *start, struct timespec *stop, struct timespec *result) {
if ((stop->tv_nsec - start->tv_nsec) < 0) {
result->tv_sec = stop->tv_sec - start->tv_sec - 1;
result->tv_nsec = stop->tv_nsec - start->tv_nsec + 1000000000;
} else {
result->tv_sec = stop->tv_sec - start->tv_sec;
result->tv_nsec = stop->tv_nsec - start->tv_nsec;
}
return;
}
int main(int argc, char** argv) {
if (geteuid() != 0) {
printf("uid is not 0 - checksudo must be owned by uid 0 and have the setuid bit set\n");
return 2;
}
struct timespec current_time;
if (clock_gettime(CLOCK_BOOTTIME, ¤t_time) != 0) {
printf("Unable to get current time: %s\n", strerror(errno));
return 2;
}
struct stat ttypath_stat;
if (stat(ttyname(0), &ttypath_stat) != 0) {
printf("Unable to stat current tty: %s\n", strerror(errno));
return 2;
}
FILE* timestamp_fd = fopen("/var/run/sudo/ts/" USERNAME, "rb");
if (timestamp_fd == NULL) {
printf("Unable to open sudo timestamp file: %s\n", strerror(errno));
return 2;
}
long offset = 0;
int found = 0;
while (1) {
if (fseek(timestamp_fd, offset, SEEK_SET) != 0) {
printf("Failed to seek timestamp file: %s\n", strerror(errno));
return 2;
}
unsigned short timestamp_entry_header[4];
if (feof(timestamp_fd)) {
printf("matching timestamp not found\n");
return 2;
}
if (fread(×tamp_entry_header, sizeof(unsigned short), 4, timestamp_fd) < 4) {
break;
}
if (ferror(timestamp_fd)) {
printf("IO error when reading timestamp file\n");
return 2;
}
// read tty device id
if (timestamp_entry_header[2] == 2 && timestamp_entry_header[3] == 0) {
if (fseek(timestamp_fd, offset + 32, SEEK_SET) != 0) {
printf("Failed to seek timestamp file: %s\n", strerror(errno));
return 2;
}
dev_t tty_dev_id;
if (fread(&tty_dev_id, sizeof(dev_t), 1, timestamp_fd) < 1) {
printf("EOF when reading tty device id\n");
return 2;
}
if (tty_dev_id == ttypath_stat.st_rdev) {
// read timestamp
if (fseek(timestamp_fd, offset + 16, SEEK_SET) != 0) {
printf("Failed to seek timestamp file: %s\n", strerror(errno));
return 2;
}
struct timespec sudo_time;
if (fread(&sudo_time, sizeof(struct timespec), 1, timestamp_fd) < 1) {
printf("EOF when reading timestamp\n");
return 2;
}
struct timespec time_since_sudo;
timespec_diff(&sudo_time, ¤t_time, &time_since_sudo);
found = time_since_sudo.tv_sec < TIMEOUT * 60;
break;
}
}
offset += timestamp_entry_header[1];
}
fclose(timestamp_fd);
return !found;
}