Nach einer Analyse des Scriptes mit einem guten Freund (welcher ein exzellenter Programmierer ist) haben wir das Script in Punkten Sicherheit und Performance etwas optimiert. Die Do-While Schleife war hier in diesem Falle etwas schlecht gewählt, da die Schleife durch das doppelt genutzte get-Random viel zu oft durch gelaufen ist. Ein Beispiel: bei einem 100 stelligen Passwort ist diese Schleife ±160 mal durchgelaufen, bei 1000 Stellen steigt dies extrem auf über ±1360. Ist dann noch definiert, dass nur große Buchstaben gewählt werden sollen ist das Script dann eher für die Tonne. Denn hier durchläuft das Script die Schleife gleich ±4000 mal. Das ist absolut nicht tragbar in Sachen Performance. Es kann im Worste Case bei falsch gerundeten Zahlen oder ähnlichem passieren, dass die Schleife eine Endlosschleife wird. Ebenfalls habe ich festgestellt das Get-Random nicht so schnell umgesetzt werden kann wie ein erzeugtes Random Objekt mittels $Random = New-Object Random. Dieses lässt sich einfacher wieder verwenden. So wird auch bspw. Random in C# aufgerufen. Grundlage ist hier das .NET Framework. Vorab sei gesagt, der Passwort Generator der Version 2 ist weniger anfällig für Falschangaben und berechnet ein Passwort schneller als die erste Version. Dennoch läuft eine Version in C# deutlich schneller. Aber dazu ein anderes Mal mehr.
Was habe ich im Script genutzt
Für dieses Script habe ich mehrere bereits beschriebene Elemente genutzt. Der Generator basiert auf mehreren Instanzen von Random. Die in Version 1 genutzte Switch- oder auch Case-Schleife (PowerShell – 11.1 – Logfiles mit einer Function erstellen) ist rausgeflogen. Ich nutze in dieser Version nur noch einfache For-Schleifen. Die benötigteZeit für das Script wird weiterhin ermittelt. Dies kann entweder mit Measure-Command (PowerShell – Scriptdauer messen Teil 2) oder mit einer selbstgeschrieben Variante (PowerShell – Scriptdauer messen Teil 1) gemessen werden. Ich habe mich im Script für meine selbst erstellte Variante entschieden. Zu guter Letzt nutze ich die Funktion Runden (Round) aus der Mathematikbibliothek (Math).
Input des Scriptes
Im Script müssen mehrere Variablen angeben werden. Diese habe ich in der folgenden Tabelle näher beschrieben.
Name | Typ | Beschreibung |
$PWDLength | Int32 | Diese Variable legt die Länge des Passwortes fest. Definiert ist sie als Int32 damit auch besonders lange Passwörter möglich sind (bis zu 2.147.483.647 Zeichen). |
$AmountABC_percent | Int | Diese Variable legt den prozentualen Wert der Großbuchstaben (A-Z) fest. Dieser Wert kann von 0 bis 100 gesetzt werden. Wobei Gesamtsummer aller $Amount* Variablen nicht über 100 sein darf. |
$AmountABCsm_percent | Int | Diese Variable legt den prozentualen Wert der Kleinbuchstaben fest (a-z). Dieser Wert kann von 0 bis 100 gesetzt werden. Wobei Gesamtsummer aller $Amount* Variablen nicht über 100 sein darf. |
$AmountNR_percent | Int | Diese Variable legt den prozentualen Wert der Ziffern(0-9) fest. Dieser Wert kann von 0 bis 100 gesetzt werden. Wobei Gesamtsummer aller $Amount* Variablen nicht über 100 sein darf. |
$AmountSign_percent | Int | Diese Variable legt den prozentualen Wert der Sonderzeichen ($,!,%,^) fest. Dieser Wert kann von 0 bis 100 gesetzt werden. Wobei Gesamtsummer aller $Amount* Variablen nicht über 100 sein darf. |
$AllAmounts_percent | Int |
Diese Variable fast alle $Amount* Variablen zusammen. Anschließend wird geprüft, ob diese größer gleich 100 ist. Ist sie größer als 100 wird ein Fehler ausgegeben mit dem Hinweis das die Werte angepasst werden müssen. Ist diese Variable kleiner 100 wird der Wert von $AmountABC_percent solange um jeweils 1 erhöht bis der Wert von $AllAmounts_percent 100 entspricht. |
Funktionsweise des Scriptes
Nachdem die oben genannten Werte angeben sind geht es im Script wie folgt weiter. Es wird zu erst geprüft, ob die Variable $AllAmounts_percent kleiner 100 ist. Trifft dies zu, wird der Wert der Variable $AmountABC_percent (Anzahl Großbuchstaben) jeweils um eins erhöht bis $AllAmounts_percent gleich 100 ist. Übersteigt $AllAmounts_percent jedoch den Wert 100 bricht das Script mit der Fehlermeldung „Prozentuale Werte sind über 100%!“ ab. Ausgelöst durch break.
Im nächsten Schritt werden vier Arrays ($Chars_*) initiiert für Groß- und Kleinbuchstaben, Ziffern sowie für Sonderzeichen. Diese können beliebig erweitert werden.
1 2 3 4 |
$Chars_ABC =@("A","B","C","D","E","F","G","H","J","K","L","M","N","P","Q","R","S","T","U","V","W","X","Y","Z") $Chars_ABCsm = @("a","b","c","d","e","f","g","h","i","j","k","m","n","o","p","q","r","s","t","u","v","w","x","y","z") $Chars_Nr = @("1","2","3","4","5","6","7","8","9") $Chars_sign = @("$","!","%","^") |
Nun werden zwei als String deklarierte Variablen $UserPassword0 und $UserPassword angelegt. Die Variable $UserPassword0 wird für eine spätere For-Schleife verwendet. Inhalt der Variable $UserPassword ist am Ende das generierte Passwort. Dazu müssen sie vorher bereits existieren.
1 |
[String]$UserPassword0=$UserPassword="" |
Es werden zwei weitere Arrays gebildet, $AllAmounts_array und $AllAmountsArray. Das Array $AllAmounts_array enthält alle vier Werte der $Amount* Variablen. $AllAmountsArray wird zunächst nur initiiert. Dieses wird für die folgende For-Schleife benötigt. Weiterhin wird eine Variable $CountAmmount gerbildet. Diese dient zur Prüfung ob die gewählte Passwortlänge erreicht wurde.
1 2 3 |
$AllAmounts_array =@($AmountABC_percent,$AmountABCsm_percent,$AmountNR_percent,$AmountSign_percent) $AllAmountsArray = @() $CountAmmount=0 |
In der folgenden For-Schleife kommt die Round Funktion der Math Bibliothek zum Einsatz. Hier breche ich die angegebenen prozentualen Werte auf Ganzahlen (Integer) herunter. Dazu nehme ich die Länge des Passwortes geteilt durch 100. Das Ergebnis wird anschließend mit dem Wert von $AllAmounts_array[$i] multipliziert. $i entspricht jeweils dem Wert von $AmountABC_percent, $AmountABCsm_percent, $AmountNR_percent oder $AmountSign_percent.
1 2 3 4 5 6 |
for($i=0;$i -lt $AllAmounts_array.Count;$i++) { $Percentage = [math]::Round($PWDLength/100*$AllAmounts_array[$i]) $AllAmountsArray += $Percentage $countAmmount += $Percentage } |
Die gesamt ermittelten Werte von $AllAmountsArray werden auf die entsprechenden $Amount*_numeric Variablen verteilt. Es kann sein das die Gesamtsumme nicht 100 erreicht. Dies kommt durch das Runden der Prozentangaben.
1 2 3 4 |
$AmountABC_numeric = $AllAmountsArray[0] $AmountABCsm_numeric = $AllAmountsArray[1] $AmountNR_numeric = $AllAmountsArray[2] $AmountSign_numeric = $AllAmountsArray[3] |
In der folgenden If-Schleife wir der Wert von $CountAmmount geprüft. Ist dieser nicht 100 wird auch hier wieder der Anteil von Großbuchstaben ($AmountABC_numeric) erhöht. Somit ist gewehrleistet, das die Passwortlänge in jedem Fall erreicht wird.
1 2 3 4 5 |
If ($CountAmmount -ne $PWDLength) { $difference = $PWDLength - $CountAmmount $AmountABC_numeric += $difference } |
Anschließend wird das Random Objekt erzeugt und die Summen der einzelnen $Chars_* Array’s in $Chars_*_count Variablen gespeichert. Diese werden anschließend in der nächsten For-Schleife verwendet. Warum werden die Summen nochmals in extra Variablen gespeichert? Der Grund liegt hier in der Performance. Wenn der Wert $Variable.Count in der For-Schleife abgerufen wird, wird dieser jedes mal erneut ausgelesen. Das geht auf den Prozessor und verursacht mehr Last. Die Variable hingegen ist bereits gesetzt und im Speicher.
1 2 3 4 5 6 |
$Random = New-Object Random $Chars_ABC_count = $Chars_ABC.Count $Chars_ABCsm_Count = $Chars_ABCsm.Count $Chars_Nr_Count = $Chars_Nr.Count $Chars_sign_Count = $Chars_sign.Count |
Nach diesen ganzem Vorgeplänkel kommt nun der eigentliche und spannende Teil, das Zusammenstellen des Passwortes. In der For-Schleife wird $i solange hochgezählt bis dieser die Länge von $PWDLength erreicht hat. In der Schleife selbst werden die einzelnen $Ammount*_numeric Werte jeweils mittels einer If-Bedingung runtergezählt solange diese größer oder gleich 0 sind. In der einzelnen If-Bedingung wird dann dem String $UserPasswort0 ein zufälliges Zeichen angehangen. Beispielsweise sieht bei einem 10 stelligen Passwort mit gleicher Verteilung der Prozentwerte (je 25%) die Variable $UserPasswort0 nach der For.-Schleife wie folgt aus: Vk7!Le0?AP.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
for($i=0;$i-lt$PWDLength;$i++) { if($AmountABC_numeric-- -gt 0) { $UserPassword0 += $Chars_ABC[$($Random.Next(0, $Chars_ABC_Count))] } if($AmountABCsm_numeric-- -gt 0) { $UserPassword0 += $Chars_ABCsm[$($Random.Next(0, $Chars_ABCsm_Count))] } if($AmountNR_numeric-- -gt 0) { $UserPassword0 += $Chars_Nr[$($Random.Next(0, $Chars_Nr_Count))] } if($AmountSign_numeric-- -gt 0) { $UserPassword0 += $Chars_sign[$($Random.Next(0, $Chars_sign_Count))] } } |
Das ist natürlich noch nicht schön anzusehen, da es hier einen sichtbaren Algorithmus gibt. Deshalb wird anschließend das Ganze nochmal „gemixt“. Dazu wird der String $Userpassword0 in ein Array $UserPasswordArr übergeben mittels der Funktion .ToCharArray().
1 |
$UserPasswordArr = $UserPassword0.ToCharArray() |
In einer For-Schleife vertausche ich einfach immer eine zufällige Stelle des Array mit einer anderen. Das geht solange bis $i gleich der Hälfte der Passwortlänge ist. Ein beliebiges Feld eines Array lässt sich unter PowerShell wie folgt füllen. $Array[2] =“Eintrag in 3.Feld des Array“. Um hier den Tausch der Variablen zu bewerkstelligen benötige ich eine Hilfsvariable $tmp. In der Hilfsvariable wird der erste Wert zwischengespeichert. Anschließend wird dieser mit dem zweiten Werten überschrieben. Der zweite Wert wird dann von der $tmp Variable überschrieben. So wurde Wert eins zu Wert zwei und umgekehrt.
1 2 3 4 5 6 7 8 9 |
for($i=0;$i -le $($UserPassword0.Length/2);$i++) { $random1 = $Random.Next(0,($UserPasswordArr.Count)) $random2 = $Random.Next(0,($UserPasswordArr.Count)) $tmp = $UserPasswordArr[$random1] $UserPasswordArr[$random1] = $UserPasswordArr[$random2] $UserPasswordArr[$random2] = $tmp } $tmp="" |
Jetzt muss ich aus dem Array $UserPasswordArr wieder ein String machen. Hierzu nutze ich einfach eine Foreach-Schleife und hänge dem String $UserPasword die einzelnen Felder des Array an.
1 2 3 4 |
foreach($Field in $UserPasswordArr) { $UserPassword += $Field } |
Anschließend folgt nur noch der Output. Hier wird zu erst das Passwort angezeigt, gefolgt von der Länge des Passwortes. Darunter wird noch die benötigte Zeit angeben die das Script gebraucht hat. Fertig 🙂
1 2 3 4 5 6 7 8 9 10 11 |
# Output Write-Host "Passwort:"-BackgroundColor Magenta $UserPassword Write-Host "Passwortlänge:"-BackgroundColor Magenta $UserPassword.Length # Benötigte Zeit $endtime = date $time = $endtime - $startime Write-Host "Benötigte Zeit zur Erstellung:"-BackgroundColor Magenta Write-Host "$($time.Hours)h:$($time.Minutes)m:$($time.Seconds)s:$($time.Milliseconds)ms" -BackgroundColor Magenta |
Weitere Features
Das Script werde ich noch als Funktion umschreiben damit die Angabe der Werte einfacher von der Hand gehen. Zudem möchte ich das Script mit einer GUI verstehen und als umgewandelte ausführbare Datei (EXE) erstellen. Sodass ein einfache Starten als Tool möglich ist. Diesen Vorgang werde ich natürlich auch hier wieder detailliert beschreiben.
Eines der größeren Schritte soll dann die Aufnahme in ein selbst erstelltes Modul werden. Das Modul soll dann mehrere praktische Scripte enthalten und stets erweitert werden. Hierzu werde ich dann eine eigene Reihe von Artikeln auf meinem Blog verfassen.
Beide Versionen habe ich auch in die DLs gepackt.
[wpfilebase tag=file id=7 /]
[wpfilebase tag=file id=8 /]
Das komplette Script
Hier nun das Script in seiner vollen Länge mit Header und Versionsverwaltung. Beispielausgaben des Scriptes stehen darunter.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 |
<# Name Simple Password Generator Beschreibung Erzeugt ein beliebig langes Passwort aus vorgegebenen Zeichen. $PWDLength ist die Passwortlänge. Es kann der Prozentuale Anteil der 4 Zeichen-Bereiche gewählt werden. Dieser darf gesamt nicht über 100 kommen. Liegt er unterhalb von 100 wird der Wert vom Array $AmountABC_percent solange erhöht bis er 100 erreicht. Version, Datum, Beschreibung, Autor 1.0 | 22.07.2014 | Initiale Version | Stefan Rehwald 1.1 | 27.10.2015 | Prozentproblem bereinigt | Stefan Rehwald 1.2 | 28.10.2015 | Performancesteigerung durch auflösen der Do-While-Schleife, .NET Random-Objekt | Stefan Rehwald, Max RAAB 1.3 | 02.11.2015 | Performancesteigerung, For-Schleife verbessert, String statt Array befüllen. | Stefan Rehwald, Max RAAB #> $startime = date [int32]$PWDLength = 11 $AmountABC_percent = 25 $AmountABCsm_percent = 25 $AmountNR_percent = 30 $AmountSign_percent = 20 $AllAmounts_percent = $AmountABC_percent + $AmountABCsm_percent + $AmountNR_percent + $AmountSign_percent If($AllAmounts_percent -lt 100) { do { $AmountABC_percent++ $AllAmounts_percent = $AmountABC_percent + $AmountABCsm_percent + $AmountNR_percent + $AmountSign_percent } while($AllAmounts_percent -ne 100) } elseif($AllAmounts_percent -gt 100) { Write-Host "Prozentuale Werte sind über 100%!" -ForegroundColor Red break } $Chars_ABC =@("A","B","C","D","E","F","G","H","J","K","L","M","N","P","Q","R","S","T","U","V","W","X","Y","Z") $Chars_ABCsm = @("a","b","c","d","e","f","g","h","i","j","k","m","n","o","p","q","r","s","t","u","v","w","x","y","z") $Chars_Nr = @("1","2","3","4","5","6","7","8","9") $Chars_sign = @("$","!","%","^") [String]$UserPassword0=$UserPassword="" [Array]$UserPasswordArr=@() , $AllAmounts_array =@($AmountABC_percent,$AmountABCsm_percent,$AmountNR_percent,$AmountSign_percent) $AllAmountsArray = @() $CountAmmount=0 for($i=0;$i -lt $AllAmounts_array.Count;$i++) { $Percentage = [math]::Round($PWDLength/100*$AllAmounts_array[$i]) $AllAmountsArray += $Percentage $countAmmount += $Percentage } $AmountABC_numeric = $AllAmountsArray[0] $AmountABCsm_numeric = $AllAmountsArray[1] $AmountNR_numeric = $AllAmountsArray[2] $AmountSign_numeric = $AllAmountsArray[3] If ($CountAmmount -ne $PWDLength) { $difference = $PWDLength - $CountAmmount $AmountABC_numeric += $difference } $Random = New-Object Random $Chars_ABC_count = $Chars_ABC.Count $Chars_ABCsm_Count = $Chars_ABCsm.Count $Chars_Nr_Count = $Chars_Nr.Count $Chars_sign_Count = $Chars_sign.Count for($i=0;$i-lt$PWDLength;$i++) { if($AmountABC_numeric-- -gt 0) { $UserPassword0 += $Chars_ABC[$($Random.Next(0, $Chars_ABC_Count))] } if($AmountABCsm_numeric-- -gt 0) { $UserPassword0 += $Chars_ABCsm[$($Random.Next(0, $Chars_ABCsm_Count))] } if($AmountNR_numeric-- -gt 0) { $UserPassword0 += $Chars_Nr[$($Random.Next(0, $Chars_Nr_Count))] } if($AmountSign_numeric-- -gt 0) { $UserPassword0 += $Chars_sign[$($Random.Next(0, $Chars_sign_Count))] } } $UserPasswordArr = $UserPassword0.ToCharArray() for($i=0;$i -le $($UserPassword0.Length/2);$i++) { $random1 = $Random.Next(0,($UserPasswordArr.Count)) $random2 = $Random.Next(0,($UserPasswordArr.Count)) $tmp = $UserPasswordArr[$random1] $UserPasswordArr[$random1] = $UserPasswordArr[$random2] $UserPasswordArr[$random2] = $tmp } $tmp="" foreach($Field in $UserPasswordArr) { $UserPassword += $Field } # Output Write-Host "Passwort:"-BackgroundColor Magenta $UserPassword Write-Host "Passwortlänge:"-BackgroundColor Magenta $UserPassword.Length # Benötigte Zeit $endtime = date $time = $endtime - $startime Write-Host "Benötigte Zeit zur Erstellung:"-BackgroundColor Magenta Write-Host "$($time.Hours)h:$($time.Minutes)m:$($time.Seconds)s:$($time.Milliseconds)ms" -BackgroundColor Magenta |
Ausgabe des Script
Mit steigender Länge des Passwortes benötigt das Script entsprechend mehr Zeit. Das Script lässt sich garantiert noch etwas performanter gestalten, aber ich denke für ein Passwort mit 100 Stellen sind 39 ms ganz in Ordnung. Aus optischen Gründen habe ich die Passwörter in der Ausgabe als String in Kochkommas gestellt. Die Kochkommas werden in der normalen Ausgabe nicht angezeigt.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 |
# $PWDLength = 20 # $AmountABC_percent = 30 # $AmountABCsm_percent = 20 # $AmountNR_percent = 28 # $AmountSign_percent = 22 #Ausgabe #Passwort: "$eN!Hkt2k983N$T4Ux%1" #Passwortlänge: 20 #Benötigte Zeit zur Erstellung: 0h:0m:0s:36ms <# ================================== #> # $PWDLength = 100 #Ausgabe #Passwort: "7m6$X$1$8o8uE34!Sc78C^NDDtioq!GgU82^S95$2W!3on4$E6GUEnm^U!vieyK%dA51Q27$%n1FNeN2Sy%%t2%!4TH!gg1e3933" #Passwortlänge: 100 #Benötigte Zeit zur Erstellung: 0h:0m:0s:47ms <# ================================== #> # $PWDLength = 1.000 #Ausgabe #Passwort: "Ck1^32M1r17Z!a2d$%7^K2^%o$4$V58^F55$!b3$Kz8!Ph6!Yn57C1%^K38%%b2!NxX$Nz2!fT8!Dr12^sV!T1f%%t!$Ko7^dQgWMHZMEx9$X%5F$yqbLt8^pn7N9d9a2D6FW5$8J!3%^o6cNC!GE15bWa7$V!8$ay$uUx!TBd^^2%y!4!54^tf%No8^CD%!de7oh5xCmr6VEu64ZG6!^%573$7!3a1JFB9!$!RtEgS$Nz5BZaa%Kx83S829Ve2 SL855wgS%1j1DCi^HNt3$9h7N5!6N8kN^Cab%Er2%^Y5$qpo6Wh!%qn$$YnfWQ^R2Aj2%Rxe$P^%%X8tkLg^hE!h$$p!MFs2^N%^%1e7MRa1!mN%VE8%!xf71kF5^9!9^e65P!P2gU6c%QE4%%P78Uv^5sXS!nU8kAKK%TwRV7^J!Pn$!%w!!Mh2!$%%1Wx2^3eE%d$96Gf3WD!7%T67UFF7WAB1rH%4s^s81nb6fYTHY2^8!C^8%Ai1U^k%Yr6 A7BD89^s5zRy93%%LiY5XHB!R6Qn!D%i4%CV6!X5!1%o9$^m2%6xS!Dc4U4$^$L84kLiY53k94%$81nA1$HK2$Z76QAs3NF4ot1qK$Yr68unmVF73$NAL^Xxv5e$3GG2R%^cMacyRHsx8YXX7$Lbk^6232^iU^JD1Q1fc8UPMeMB!3Dv21wM9YLd3818%izuZ^Un7!$%4!Yy1%BE18L57Hko%TCy2jQf7!1b286c%%W5634Y6$ABn7M!5^8a9fY u1NF13$1j4$LY!dEi$U3E6%4%65PF7DFe4^Cs!7u%SR%cjDQ2379$L2S28!!GzD$3D9%PQ$2!!^H1%L!TH!HE2Z6$18$V$%69G5hN7wW^FS57L4YVc1M$$7!NSBUoA3o3C8XbKvP%R^$!F8D%6ZW9CTHu^T$7G3F995Q9LASi3f%Ej7%8Sh43RTaj$2A985!2B5P772F7H59doSH14!MHS2%ZA95FE!6QAst1QiP6GT" #Passwortlänge: 1000 #Benötigte Zeit zur Erstellung: 0h:0m:0s:84ms <# ================================== #> # $PWDLength = 10.000 #Ausgabe #Passwort: # Aus Platzgründen weggelassen. #Passwortlänge: 1000 #Benötigte Zeit zur Erstellung: 0h:0m:0s:537ms <# ================================== #> # $PWDLength = 100.000 #Ausgabe #Passwort: # Aus Platzgründen weggelassen. #Passwortlänge: 100000 Benötigte Zeit zur Erstellung: 0h:0m:18s:973ms |
Viele Grüße
rewe
Ein Gedanke zu „PowerShell – 13.1 – Simpler Passwortgenerator Version 2“