aboutsummaryrefslogtreecommitdiffstats
path: root/go
diff options
context:
space:
mode:
authorDennis Eriksen <d@ennis.no>2023-09-05 09:33:42 +0200
committerDennis Eriksen <d@ennis.no>2023-09-05 09:33:42 +0200
commita5b608b3f120dca3f1fa2e44e7631aabdfbd1bc5 (patch)
tree63efd1d01bb5828d008293634f13992f5398a65d /go
parentadding licence (diff)
downloadmakepass-a5b608b3f120dca3f1fa2e44e7631aabdfbd1bc5.tar.gz
adding golang-version
Diffstat (limited to '')
-rw-r--r--go/go.mod7
-rw-r--r--go/go.sum4
-rw-r--r--go/makepass.go171
3 files changed, 182 insertions, 0 deletions
diff --git a/go/go.mod b/go/go.mod
new file mode 100644
index 0000000..bc5ed12
--- /dev/null
+++ b/go/go.mod
@@ -0,0 +1,7 @@
+module makepass
+
+go 1.20
+
+require golang.org/x/term v0.12.0
+
+require golang.org/x/sys v0.12.0 // indirect
diff --git a/go/go.sum b/go/go.sum
new file mode 100644
index 0000000..8c01403
--- /dev/null
+++ b/go/go.sum
@@ -0,0 +1,4 @@
+golang.org/x/sys v0.12.0 h1:CM0HF96J0hcLAwsHPJZjfdNzs0gftsLfgKt57wWHJ0o=
+golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/term v0.12.0 h1:/ZfYdc3zq+q02Rv9vGqTeSItdzZTSNDmfTi0mBAuidU=
+golang.org/x/term v0.12.0/go.mod h1:owVbMEjm3cBLCHdkQu9b1opXd4ETQWc3BhuQGKgXgvU=
diff --git a/go/makepass.go b/go/makepass.go
new file mode 100644
index 0000000..542b63e
--- /dev/null
+++ b/go/makepass.go
@@ -0,0 +1,171 @@
+package main
+
+import (
+ "bufio"
+ "fmt"
+ "golang.org/x/term"
+ "math/rand"
+ "os"
+ "strconv"
+ "strings"
+ "time"
+)
+
+// Basic constant definitions
+const (
+ normNum = 10 // Number of normal passwords
+ specNum = 6 // Number of special passwords
+ passNum = 6 // Number of passphrases
+
+ maxLength = 255 // Maximum length of passwords
+ rangeMax = 42 // Maximum range for password length
+ rangeMin = 8 // Minimum range for password length
+ passWords = 8 // Number of words in passphrases
+
+ lower = "abcdefghijklmnopqrstuvwxyz"
+ upper = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
+ digit = "0123456789"
+ other = "!$%&#/()=?+-_,.;:<>[]{}|@*" // Special characters for special passwords
+
+ defaultFile = "/usr/share/dict/words" // Default path to the file of words used for passphrases
+)
+
+// Variables for different types of passwords
+var (
+ alpha []string = strings.Split(lower+upper, "") // Array of alphabets (lowercase and uppercase)
+ alnum []string = append(alpha, strings.Split(digit, "")...) // Array of alphanumeric characters
+ every []string = append(alnum, strings.Split(other, "")...) // Array of all characters (alphabet, digit and special)
+ normal []string = append(alnum, "-", "_") // Characters for normal passwords
+ special []string = every // Characters for special passwords
+ wordlist string = defaultFile // Path to a dictionary to use for passphrases
+ length int = 0 // Length of passwords
+)
+
+func main() {
+ //
+ // Error handling
+ //
+ var err error
+
+ // Checking command line argument and environment variable for password length
+ if len(os.Args) == 2 {
+ length, err = strconv.Atoi(os.Args[1])
+ } else if len(os.Getenv("MAKEPASS_DEFAULT_LENGTH")) > 0 {
+ length, err = strconv.Atoi(os.Getenv("MAKEPASS_DEFAULT_LENGTH"))
+ }
+
+ // If there is an error in conversion, or if the length is not within limits, exit the program with an error message
+ if err != nil || (length > maxLength || length < 0) {
+ fmt.Printf("Error: Argument must be a whole number between 0 and %d.\n", maxLength)
+ os.Exit(1)
+ }
+
+ // Get wordlist from env
+ if len(os.Getenv("MAKEPASS_WORDLIST")) > 0 {
+ wordlist = os.Getenv("MAKEPASS_WORDLIST")
+ }
+
+ // Check if the wordlist file exists
+ if _, err := os.Stat(wordlist); os.IsNotExist(err) {
+ fmt.Println("The file", wordlist, "does not exist")
+ os.Exit(1)
+ }
+
+ //
+ // move on
+ //
+
+ // initialise the random seed
+ // TODO: Get seed from /dev/random, like we do in the perl- and zsh-versions?
+ rand.Seed(time.Now().UnixNano())
+
+ // get screen width
+ termWidth, _, err := term.GetSize(0)
+
+ // Read wordlist from file
+ file, err := os.Open(wordlist)
+ if err != nil {
+ fmt.Printf("failed to open file: %s", err)
+ }
+ scanner := bufio.NewScanner(file)
+ scanner.Split(bufio.ScanLines)
+ var text []string
+ for scanner.Scan() {
+ text = append(text, scanner.Text())
+ }
+ file.Close()
+
+ //
+ // print passwords
+ //
+
+ // Generate and print normal and special passwords
+ printColumns("Normal passwords:", 10, normal, termWidth)
+ fmt.Println("")
+ printColumns("Passwords with special characters:", 6, special, termWidth)
+ fmt.Println("")
+
+ // Generate and print passphrases
+ for i := 0; i < passNum; i++ {
+ fmt.Println(passphrase(&text))
+ }
+}
+
+// Function to generate a random string of given length from given characters
+func randstring(chars []string) string {
+ l := length
+ if length == 0 { // Random length if not specified
+ l = rand.Intn(rangeMax-rangeMin) + rangeMin
+ }
+ var str strings.Builder
+
+ // Add random characters to str
+ for i := 1; i <= l; i++ {
+ if i == 1 || i == l { // don't want passwords to start or end with special characters
+ str.WriteString(alnum[rand.Intn(len(alnum))])
+ } else {
+ str.WriteString(chars[rand.Intn(len(chars))])
+ }
+ }
+ return str.String()
+}
+
+// Function to generate and print passwords
+func printColumns(title string, num int, chars []string, termWidth int) {
+
+ // column width
+ colWidth := rangeMax + 2
+ if length > 0 {
+ colWidth = length + 2
+ }
+
+ // number of colums
+ columns := termWidth / colWidth
+ if columns == 0 {
+ columns = 1
+ }
+
+ // Print title
+ fmt.Println(title)
+
+ // Print passwords in columns
+ for i := 1; i <= num; i++ {
+ fmt.Printf("%-[1]*[2]s", colWidth, randstring(chars))
+ if i%columns == 0 || (i == num && i%columns > 0) { // Add newlines
+ fmt.Println("")
+ }
+ }
+}
+
+// Function to generate passphrases
+func passphrase(arrh *[]string) string {
+ text := *arrh
+ var str strings.Builder
+ for i := 0; i < passWords; i++ {
+ str.WriteString(text[rand.Intn(len(text))]) // Write a random word to the passphrase
+ if i != passWords-1 {
+ str.WriteString("-") // Add hyphen between words
+ }
+ }
+ return str.String()
+}