solaris10(x86)のOSでユーザーパスワードを自動変更するためのスクリプトを作成しています。
※使用言語はperlでExpectモジュールを使っています。
仕様としては以下のパスワード制限を満たす文字列を自動生成し、
リストファイル(list.lst)を元にExpectモジュールを使って、
ユーザーパスワードを自動変更後に結果をファイル(result.log)へ出力します。
パスワード制限
・アルファベット大文字小文字(1文字以上)含む
・数字を(1文字以上)含む
・記号を(1文字以上)含む
ですが、実際に変更しようとすると、
稀に記号を含まないパスワードが出来てしまいます。
以下の場合は「test1」ユーザーがそれに当たります。
# cat result.log
User: test1
password: 14P2YiZy
User: test2
password: @u1iXr%q
User: test3
password: EQ2t6H(6
rand関数の使い方が良くないのか、
正常に記号部分の配列にpushされていないのか、
正直お手上げです。
以下にスクリプトの内容を記載しますので、
記号含むパスワードを自動生成するにはどうしたらよいか、
どなたかアドバイス頂けませんでしょうか。
宜しくお願い致します。
#--------------------------------
#!/usr/bin/perl -w
use strict;
use File::Copy;
use Expect;
##### password configuration #####
my $lng = 8; #Length
my $num = 1; #Number (0=disabled, 1=enabled)
my $cap = 1; #Upper (0=disabled, 1=enabled)
my $sml = 1; #Lower (0=disabled, 1=enabled)
my $mark = 1; #Symbol (0=disabled, 1=enabled)
my ($sec, $min, $hour, $mday, $mon, $year, $wday, $yday, $isdst) = localtime(time);
$mon += 1;
$year += 1900;
##### Variable initialization #####
my $ps;
my @words;
##### log files #####
my $LGFILE = 'result.log';
my $LGFILE_OLD = "result.log.${year}${mon}${mday}";
##### file list PATH #####
my $file='list.lst';
my $SADW = '/etc/shadow';
if ($num == 1) {
push @words, (0..9);
}
if ($cap == 1) {
push @words, ('A'..'Z');
}
if ($sml == 1) {
push @words, ('a'..'z');
}
if ($mark == 1) {
push @words, ('#', '$', '%', '(', ')', '-', '=',
'^', '~', '|', '@', '`', '[', ';', ':', '+', '*',
'/', '?', '_');
}
if (-e "$LGFILE_OLD"){
exit(1);
}
copy "$SADW", "$SADW.${year}${mon}${mday}" or die "Copy failure $!";
chmod(0400, "$SADW.${year}${mon}${mday}");
if (-e "$LGFILE"){
move("$LGFILE", "$LGFILE_OLD");
chmod(0400, "$LGFILE_OLD");
}
open(F, "<$file") or die "Cannot open $file: $!";
while( my $line = readline F ){
chomp($line);
open(IN, ">>$LGFILE");
for (my $var = 0; $var < $lng; $var++) {
my $words_num = @words;
$ps = $ps . "@words[rand($words_num)]";
}
my $command = Expect->spawn("passwd ${line}") or die "Can't start program... $!\n";
$command->log_stdout(0);
$command->expect(1, "New password:");
print $command "$ps\n";
$command->expect(1, "Re-enter new password:");
print $command "$ps\n";
print IN "User:\t${line}\n";
print IN "password:\t${ps}\n";
$command->soft_close();
undef $ps;
chmod(0400, "$LGFILE");
close(IN);
}
close(F);
exit(0);
#--------------------------------
No.2ベストアンサー
- 回答日時:
このプログラムがしているのは、パスワードとして利用可能な文字を@wordsに登録し、その中からランダムに文字を選ぶということですよね?これだと、記号が選ばれる可能性はあっても、必ず記号が選ばれるかは保証されません。
なぜなら、期待する条件を満たすかをチェックするコードは一切無いからです。つまり、
for (my $var = 0; $var < $lng; $var++) {
my $words_num = @words;
$ps = $ps . "@words[rand($words_num)]";
}
という、パスワードを生成しているコードを下記のようなコードに書き換える必要があるでしょう。
while (1) {
for (my $var = 0; $var < $lng; $var++) {
my $words_num = @words;
$ps = $ps . "@words[rand($words_num)]";
}
条件を満たすかチェックし、満たした場合のみbreak, 満たさない場合は$psを''でクリア。
}
$num などの条件を一切無視すると、チェックするコードはこんな感じですね。
if ($ps =~ /[0-9]/ && $ps =~ /[a-z]/ && $ps =~ /[A-Z]/ && $ps =~ /[#$%()-=略]/) { break; }
else { $ps =''; }
ご回答ありがとうございます。
無限ループ処置のなかに判定部分を入れて、
制限を満たさない場合は、ループを繰り返すようにするんですね。
大変勉強になりました。
ベストアンサーとさせて頂きます。
No.1
- 回答日時:
>@words[ rand($words_num) ]
結局のところ、上記のように、@wordsのうちの1文字を単にランダムで選んでるだけなので、「hogehoge」みたいのが生成される可能性はあるかと。
・アルファベット小文字をn文字(n=1~5)
・アルファベット大文字をm文字(m=1~6-n)
・数字をl文字(l=1~7-n-m)
・記号をk文字(k=8-n-m-l)
――と選んで、それの順列の一つを選ぶみたいな感じにするとか。
参考URL:http://ideone.com/abdnd
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・漫画をレンタルでお得に読める!
- ・街中で見かけて「グッときた人」の思い出
- ・「一気に最後まで読んだ」本、教えて下さい!
- ・幼稚園時代「何組」でしたか?
- ・激凹みから立ち直る方法
- ・1つだけ過去を変えられるとしたら?
- ・【あるあるbot連動企画】あるあるbotに投稿したけど採用されなかったあるある募集
- ・【あるあるbot連動企画】フォロワー20万人のアカウントであなたのあるあるを披露してみませんか?
- ・映画のエンドロール観る派?観ない派?
- ・海外旅行から帰ってきたら、まず何を食べる?
- ・誕生日にもらった意外なもの
- ・天使と悪魔選手権
- ・ちょっと先の未来クイズ第2問
- ・【大喜利】【投稿~9/7】 ロボットの住む世界で流行ってる罰ゲームとは?
- ・推しミネラルウォーターはありますか?
- ・都道府県穴埋めゲーム
- ・この人頭いいなと思ったエピソード
- ・準・究極の選択
- ・ゆるやかでぃべーと タイムマシンを破壊すべきか。
- ・歩いた自慢大会
- ・許せない心理テスト
- ・字面がカッコいい英単語
- ・これ何て呼びますか Part2
- ・人生で一番思い出に残ってる靴
- ・ゆるやかでぃべーと すべての高校生はアルバイトをするべきだ。
- ・初めて自分の家と他人の家が違う、と意識した時
- ・単二電池
- ・チョコミントアイス
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
close()で例外が投げられる理由
-
Perl で syntax error
-
改行コードが勝手に
-
ExcelをCSV書き出す場合のシー...
-
エクセルVBA コードが同じでも...
-
vba dir の相対パス
-
batファイルでrenameができませ...
-
VBAでCSVファイルを途中行まで...
-
window.open でのファイル指定方法
-
バッチファイルの作り方(CSV→...
-
XML::Parserの parsefileの結果...
-
VBAで巨大なファイルの途中から...
-
VBAでCSVファイルの特定行を書...
-
fgets で値が取得できない
-
ADOによるCSVファイルからのデ...
-
タブの色を変更する方法
-
DOSコマンドで、標準出力を出力...
-
圧縮された.tgzファイルの一部...
-
system関数と引数について
-
FORTRANのプログラミング
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
close()で例外が投げられる理由
-
改行コードが勝手に
-
オープンしたファイルで行の連結
-
パスワードを外部ファイルから...
-
perlで大容量CSVのsort方法につ...
-
die関数のエラー出力先について
-
while(<ハンドラ>) {} で行数を...
-
Perl で syntax error
-
巨大ファイルの行をを逆順に並...
-
perlで複数のファイルの処理に...
-
where can I buy snowbord in t...
-
ログファイルを編集する方法は...
-
テキストファイルの本文中に行...
-
Perlでのファイル一括読み込み
-
Perl:ファイルハンドルをスカラ...
-
cgiの投票回数制限設定について...
-
GD.pmで作成した画像を保存する...
-
perlで、文字列の中から何番目...
-
perl 初心者です。 わかりや...
-
クリックを何回もされて、重複...
おすすめ情報