aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--README.md53
-rwxr-xr-xmakepass.py318
2 files changed, 349 insertions, 22 deletions
diff --git a/README.md b/README.md
index d1dc0b4..1abd74c 100644
--- a/README.md
+++ b/README.md
@@ -163,36 +163,41 @@ Here are the results so far:
```
% hyperfine --time-unit=millisecond --warmup=1 --shell=none ./makepass.*
Benchmark 1: ./makepass.bash
- Time (mean ± σ): 77.4 ms ± 2.1 ms [User: 27.8 ms, System: 33.8 ms]
- Range (min … max): 72.8 ms … 80.8 ms 37 runs
+ Time (mean ± σ): 81.2 ms ± 1.9 ms [User: 34.0 ms, System: 39.7 ms]
+ Range (min … max): 77.3 ms … 85.2 ms 35 runs
Benchmark 2: ./makepass.go
- Time (mean ± σ): 7.2 ms ± 0.2 ms [User: 1.2 ms, System: 4.9 ms]
- Range (min … max): 6.6 ms … 7.6 ms 409 runs
+ Time (mean ± σ): 6.7 ms ± 0.1 ms [User: 1.0 ms, System: 4.9 ms]
+ Range (min … max): 6.4 ms … 7.8 ms 444 runs
Benchmark 3: ./makepass.pl
- Time (mean ± σ): 39.4 ms ± 0.3 ms [User: 19.6 ms, System: 15.4 ms]
- Range (min … max): 38.8 ms … 40.4 ms 76 runs
+ Time (mean ± σ): 40.4 ms ± 0.3 ms [User: 19.7 ms, System: 18.3 ms]
+ Range (min … max): 39.7 ms … 41.3 ms 75 runs
-Benchmark 4: ./makepass.rs
- Time (mean ± σ): 4.8 ms ± 0.2 ms [User: 1.3 ms, System: 2.3 ms]
- Range (min … max): 4.3 ms … 5.7 ms 606 runs
+Benchmark 4: ./makepass.py
+ Time (mean ± σ): 103.5 ms ± 1.2 ms [User: 71.0 ms, System: 27.9 ms]
+ Range (min … max): 101.9 ms … 107.8 ms 29 runs
-Benchmark 5: ./makepass.sh
- Time (mean ± σ): 646.5 ms ± 2.6 ms [User: 145.0 ms, System: 1005.0 ms]
- Range (min … max): 643.2 ms … 651.3 ms 10 runs
+Benchmark 5: ./makepass.rs
+ Time (mean ± σ): 4.9 ms ± 0.2 ms [User: 1.1 ms, System: 2.4 ms]
+ Range (min … max): 4.4 ms … 5.6 ms 610 runs
-Benchmark 6: ./makepass.zsh
- Time (mean ± σ): 26.9 ms ± 0.6 ms [User: 12.4 ms, System: 12.3 ms]
- Range (min … max): 25.5 ms … 28.4 ms 111 runs
+Benchmark 6: ./makepass.sh
+ Time (mean ± σ): 638.4 ms ± 2.9 ms [User: 155.0 ms, System: 915.0 ms]
+ Range (min … max): 634.0 ms … 643.9 ms 10 runs
+
+Benchmark 7: ./makepass.zsh
+ Time (mean ± σ): 26.9 ms ± 0.7 ms [User: 13.4 ms, System: 12.0 ms]
+ Range (min … max): 25.5 ms … 30.2 ms 109 runs
Summary
- './makepass.rs' ran
- 1.49 ± 0.06 times faster than './makepass.go'
- 5.56 ± 0.23 times faster than './makepass.zsh'
- 8.16 ± 0.28 times faster than './makepass.pl'
- 16.02 ± 0.69 times faster than './makepass.bash'
- 133.85 ± 4.52 times faster than './makepass.sh'
+ ./makepass.rs ran
+ 1.36 ± 0.07 times faster than ./makepass.go
+ 5.44 ± 0.31 times faster than ./makepass.zsh
+ 8.19 ± 0.42 times faster than ./makepass.pl
+ 16.46 ± 0.91 times faster than ./makepass.bash
+ 20.96 ± 1.08 times faster than ./makepass.py
+ 129.36 ± 6.50 times faster than ./makepass.sh
```
## Versions
@@ -210,6 +215,9 @@ $ go build -C go -o .. -ldflags "-s -w"
### Perl
Perl version. I like this version <3
+### Python
+I don't really care for python, but here it is. It's pretty slow, but I don't really know Python. It should probably be faster.
+
### Rust
Rust-version. It's fast!
@@ -224,4 +232,5 @@ $ cargo build --manifest-path rust/Cargo.toml --release && mv rust/target/releas
### Zsh
This is currently the "main" version. It uses pure zsh, with no forking out to
-other programs. As of adding the go-version, it is the second fastest version.
+other programs. Aside from the compiled languages, it is by far the fastest
+version.
diff --git a/makepass.py b/makepass.py
new file mode 100755
index 0000000..233ae7d
--- /dev/null
+++ b/makepass.py
@@ -0,0 +1,318 @@
+#!/usr/bin/env python3
+#
+# Author : Dennis Eriksen <d@ennis.no>
+# File : makepass.py
+# Created : 2023-09-14
+# License : BSD-3-Clause
+#
+# Copyright (c) 2018-2023 Dennis Eriksen <d@ennis.no>
+
+import argparse
+import os
+import random
+import sys
+
+# Global variables
+
+MAX = 255
+RANGE_MAX = 42
+RANGE_MIN = 8
+PASS_WORDS = 8
+
+LOWER = "abcdefghijklmnopqrstuvwxyz"
+UPPER = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
+DIGIT = "0123456789"
+OTHER = "!$%&#/()=?+-_,.;:<>[]{}|@*"
+
+ALPHA = LOWER + UPPER
+ALNUM = ALPHA + DIGIT
+EVERY = ALNUM + OTHER
+
+NORMAL = ALNUM + "_-"
+SPECIAL = EVERY
+
+DEFAULT_LENGTH = 0
+DEFAULT_NUMBER = 10
+DEFAULT_WORDLIST = "/usr/share/dict/words"
+
+
+class CommonPrintData:
+ def __init__(self, length, printbool, printlen, col_width, col_num):
+ self.length = length
+ self.printbool = printbool
+ self.printlen = printlen
+ self.col_num = col_num
+ self.col_width = col_width
+
+
+#
+# Main function
+#
+def main():
+ normal = os.getenv("MAKEPASS_NORMAL") or NORMAL
+ special = os.getenv("MAKEPASS_SPECIAL") or SPECIAL
+ wordlist_file = os.getenv("MAKEPASS_WORDLIST") or DEFAULT_WORDLIST
+
+ # Initialize parser
+ parser = argparse.ArgumentParser(
+ description="Process some strings.", add_help=False
+ )
+
+ # Add arguments
+ parser.add_argument("-h", "--help", action="store_true")
+ parser.add_argument("-l", "--length", type=int, help="Length of strings")
+ parser.add_argument("-n", "--number", type=int, help="Number of strings")
+ parser.add_argument("-p", "--printlen", action="store_true")
+ parser.add_argument("arg_length", type=int, nargs="?")
+
+ # Parse arguments
+ args = parser.parse_args()
+
+ if args.help:
+ usage()
+
+ # Some error-checking
+ # length
+ try:
+ # env or default
+ length = int(os.getenv("MAKEPASS_LENGTH") or DEFAULT_LENGTH)
+
+ # -l flag
+ if args.length is not None:
+ if args.length < 0 or MAX < args.length:
+ raise ValueError("-l")
+ length = args.length
+
+ # solo argument
+ if args.arg_length is not None:
+ length = args.arg_length
+
+ # check result
+ if length < 0 or MAX < length:
+ raise ValueError("length")
+ except ValueError as e:
+ print(f"{e} must be a number between 0 and {MAX}")
+ sys.exit(1)
+
+ # number
+ try:
+ # env or default
+ number = int(os.getenv("MAKEPASS_NUMBER") or DEFAULT_NUMBER)
+
+ # -n flag
+ if args.number is not None:
+ if args.number < 1 or MAX < args.number:
+ raise ValueError("-n")
+ number = args.number
+
+ # check result
+ if number < 1 or MAX < number:
+ raise ValueError("number")
+ except ValueError as e:
+ print(f"{e} must be a number between 1 and {MAX}")
+ sys.exit(1)
+
+ # Set some other stuff
+
+ printbool = args.printlen
+ if printbool:
+ printlen = 3 if length < 100 else 4
+ else:
+ printlen = 0
+
+ # get terminal width
+ try:
+ xwidth = os.get_terminal_size().columns
+ except Exception as _:
+ xwidth = 1
+
+ # get width of columns to print
+ if length == 0:
+ col_width = RANGE_MAX + 2
+ else:
+ col_width = length + 2
+
+ # number of columns to print
+ col_num = int(xwidth / (col_width + printlen))
+ if col_num == 0:
+ col_num = 1
+
+ # config
+ config = CommonPrintData(
+ length=length,
+ printbool=printbool,
+ printlen=printlen,
+ col_width=col_width,
+ col_num=col_num,
+ )
+
+ #
+ # print passwords
+ #
+
+ # Generate and print normal and special passwords
+ print_columns("Normal passwords", number, normal, config)
+
+ print()
+
+ print_columns(
+ "Passwords with special characters", int(number / 3 * 2 + 1), special, config
+ )
+
+ # Generate and print passphrases if wordlist exists
+ if os.path.isfile(wordlist_file):
+ print()
+
+ print("Generating passphrases: ")
+ passphrase(int(number / 2), wordlist_file)
+
+
+#
+# Print passwords in neat columns
+#
+def print_columns(title: str, num: int, chars: str, c: CommonPrintData):
+ # Generate random strings
+ strings = [randstring(c.length, chars) for _ in range(num)]
+
+ print(f"{title}:")
+
+ for i in range(num):
+ # Print the length of the string if printlen is set
+ if c.printbool:
+ print(f"{len(strings[i]):0{c.printlen-1}}", end=" ")
+
+ # Print the string, left justified based on column width
+ print(f"{strings[i]:<{c.col_width}}", end="")
+
+ # If we have printed enough strings for a single line or the
+ # rest of strings are fewer than column number, we break lines.
+ i += 1
+ if i % c.col_num == 0 or (i == num and i % c.col_num > 0):
+ print()
+
+
+#
+# Generate random strings that can work as passwords
+#
+def randstring(length, chars):
+ """
+ Generate a random password of the specified length. Use special
+ characters if special_chars option is True.
+ """
+ if length == 0:
+ length = random.randint(RANGE_MIN, RANGE_MAX)
+
+ password = []
+ password.append(random.choice(ALPHA))
+ for i in range(length - 2):
+ password.append(random.choice(chars))
+ password.append(random.choice(ALNUM))
+
+ # Randomly reorder the characters
+ return "".join(password)
+
+
+#
+# Passphrases!
+#
+def passphrase(number, file_name):
+ """
+ Generate a number of random passphrases from the wordlist.
+ """
+ try:
+ with open(file_name, "r") as f:
+ lines = f.read().splitlines()
+ except IOError:
+ print(f"Cannot open file: {file_name}")
+ sys.exit(1)
+
+ for i in range(number):
+ word_list = random.choices(lines, k=5)
+ passphrase = "-".join(word_list)
+ print(passphrase)
+
+
+def usage():
+ text = """NAME
+ makepass - create several random passwords
+
+SYNOPSIS
+ makepass [OPTIONS] [NUM]
+
+ If a NUM is provided, passwords will be NUM characters long.
+
+ By default `makepass` will output passwords from the three following classes:
+
+ - Normal passwords - random strings with letters (both lower and upper
+ case), numbers, and dashes and underscores.
+
+ - Passwords with special characters - random strings generated from lower
+ and upper case letters, numbers, and the following characters:
+ !#$%&/()=?+-_,.;:<>[]{}|@*
+
+ - Passphrases - if we find a dictionary, a series of eight random words
+ from the dictionary, separated by dashes. The number of words can not be
+ changed, but you do not have to use all of them. Use as mane as you want.
+
+ The first character will always be alphabetic, and the last will always be
+ alphanumeric.
+
+DESCRIPTION
+ makepass has the following options:
+
+ -h
+ output this help-text
+ -l
+ length of passwords. See MAKEPASS_LENGTH below
+ -n
+ number of passwords. See MAKEPASS_NUMBER below
+ -p
+ print length of number
+
+ENVIRONMENT
+ makepass examines the following environmental variables.
+
+ MAKEPASS_LENGTH
+ Specifies the length of passwords. Valid values are 0-255. If 0, a
+ random value between 8 and 42 will be used for each password. -l
+ overrides this environmental variable, and the argument NUM overrides
+ that again. So `MAKEPASS_LENGTH=10 makepass -l 12 14` will give
+ passwords that are 14 characters long, even though both -l and
+ MAKEPASS_LENGTH also specifies a length.
+
+ MAKEPASS_NUMBER
+ The number of passwords to generate. This formula is used to determine
+ how many passwords from each group should be generated:
+ - (n) normal passwords
+ - (n / 3 * 2 + 1) special passwords
+ - (n / 2) passphrases
+ Where n is 10 by default. Valid values for n are 1-255. Floating-poing
+ math is not used, so results may vary.
+
+ MAKEPASS_PRINTLEN
+ If 1, print length of all passwords. If 0, don\'t.
+
+ MAKEPASS_NORMAL
+ String of characters from which to generate "normal" passwords.
+ Defaults to:
+ abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-_
+
+ MAKEPASS_SPECIAL
+ String of characters from which to generate passwords with special
+ characters. Defaults to the same characters as in MAKEPASS_NORMAL, plus
+ these:
+ !#$%&/()=?+-_,.;:<>[]{}|@*
+
+ MAKEPASS_WORDLIST
+ Specifies the dictionary we find words for passphrases in. If this is
+ unset or empty, we try "/usr/share/dict/words". If that file does not
+ exist, no passphrases will be provided.
+
+AUTHOR
+ Dennis Eriksen <https://dnns.no>"""
+ print(text)
+ sys.exit(0)
+
+
+if __name__ == "__main__":
+ main()