Commit Diff


commit - /dev/null
commit + 5fa017d1765c7213d929cd726ebf4b2e06470cea
blob - /dev/null
blob + d20041fd4309f532c179f40839c8b1d56000a5e4 (mode 644)
--- /dev/null
+++ README.md
@@ -0,0 +1,5 @@
+# pass2vault
+
+Requires `jq` and `ksh`.
+
+**DISCLAIMER**: Vibe-coded with an LLM. I'm not much of a programmer...
blob - /dev/null
blob + 5f6f0d36ca09c492c078203fa82d645100d8d069 (mode 644)
--- /dev/null
+++ pass2vault.sh
@@ -0,0 +1,181 @@
+#!/usr/local/bin/ksh
+# Copyright (c) 2026 Manuel Kuklinski
+#
+# Permission to use, copy, modify, and distribute this software for any
+# purpose with or without fee is hereby granted, provided that the above
+# copyright notice and this permission notice appear in all copies.
+#
+# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+#
+# Import password-store entries into HashiCorp Vault (KV v2)
+# Usage: ./import.ksh <folder> (e.g. www, local, creds)
+
+if [ -z "$1" ]; then
+  echo "Usage: $0 <folder> (e.g. www, local, creds)"
+  exit 1
+fi
+
+FOLDER="$1"
+STORE="$HOME/.password-store/$FOLDER"
+export GPG_TTY=$(tty)
+TMPBLOCKS=$(mktemp /tmp/vault_blocks.XXXXXX)
+TMPARGS=$(mktemp /tmp/vault_import.XXXXXX)
+TMPSKIP=$(mktemp /tmp/vault_skip.XXXXXX)
+TMPUNKNOWN=$(mktemp /tmp/vault_unknown.XXXXXX)
+
+if [ ! -d "$STORE" ]; then
+  echo "Error: $STORE does not exist"
+  exit 1
+fi
+
+find "$STORE" -type f | while read f; do
+  rel="${f#$STORE/}"
+  entrykey="${rel%.gpg}"
+  entrykey="${entrykey%.txt}"
+
+  filetype=$(file -b "$f")
+  case "$filetype" in
+    *GPG*|*PGP*)
+      content=$(gpg --quiet --yes --batch --default-recipient-self --decrypt "$f" 2>/dev/null)
+      ;;
+    *text*|*ASCII*)
+      content=$(cat "$f")
+      ;;
+    *)
+      printf 'UNKNOWN FILE TYPE: %s (%s)\n' "$f" "$filetype" >> "$TMPSKIP"
+      continue
+      ;;
+  esac
+
+  if [ -z "$content" ]; then
+    printf 'EMPTY: %s\n' "$f" >> "$TMPSKIP"
+    continue
+  fi
+
+  > "$TMPARGS"
+  > "$TMPUNKNOWN"
+  > "$TMPBLOCKS"
+
+  # Split content into blocks separated by empty lines
+  block=""
+  while IFS= read -r line; do
+    if [ -z "$line" ]; then
+      if [ -n "$block" ]; then
+        printf '%s\n---BLOCK---\n' "$block" >> "$TMPBLOCKS"
+        block=""
+      fi
+    else
+      block="${block:+$block
+}$line"
+    fi
+  done <<< "$content"
+  [ -n "$block" ] && printf '%s\n---BLOCK---\n' "$block" >> "$TMPBLOCKS"
+
+  # Classify and process blocks
+  pw_i=1
+  login_i=1
+  current_block=""
+
+  while IFS= read -r line; do
+    if [ "$line" = "---BLOCK---" ]; then
+      [ -z "$current_block" ] && continue
+
+      # First line = password candidate
+      bpw=$(printf '%s' "$current_block" | sed -n '1p')
+      rest=$(printf '%s' "$current_block" | tail -n +2)
+
+      # Check if block contains known fields
+      has_known=$(printf '%s' "$rest" | grep -cE '^[A-Za-z][A-Za-z0-9_]*:')
+
+      if [ -z "$rest" ] || [ "$has_known" -eq 0 ]; then
+        # No known fields → custom
+        while IFS= read -r l; do
+          [ -n "$l" ] && printf '%s\n' "$l" >> "$TMPUNKNOWN"
+        done <<EOF
+$current_block
+EOF
+      else
+        # Block with known fields → password + fields
+        if [ $pw_i -eq 1 ]; then
+          printf 'password=%s\n' "$bpw" >> "$TMPARGS"
+        else
+          printf 'password%d=%s\n' "$pw_i" "$bpw" >> "$TMPARGS"
+        fi
+        pw_i=$((pw_i+1))
+
+        while IFS= read -r line2; do
+          case "$line2" in
+            [Ll]ogin:*)
+              value=$(printf '%s' "$line2" | sed 's/^[^:]*: *//')
+              if [ $login_i -eq 1 ]; then
+                printf 'login=%s\n' "$value" >> "$TMPARGS"
+              else
+                printf 'login%d=%s\n' "$login_i" "$value" >> "$TMPARGS"
+              fi
+              login_i=$((login_i+1))
+              ;;
+            [A-Za-z]*:*)
+              fkey=$(printf '%s' "$line2" | sed 's/:.*//' | tr '[:upper:]' '[:lower:]' | tr ' ' '_')
+              value=$(printf '%s' "$line2" | sed 's/^[^:]*: *//')
+              printf '%s=%s\n' "$fkey" "$value" >> "$TMPARGS"
+              ;;
+            *)
+              [ -z "$line2" ] && continue
+              printf '%s\n' "$line2" >> "$TMPUNKNOWN"
+              ;;
+          esac
+        done <<EOF
+$rest
+EOF
+      fi
+
+      current_block=""
+    else
+      current_block="${current_block:+$current_block
+}$line"
+    fi
+  done < "$TMPBLOCKS"
+
+  if [ -s "$TMPUNKNOWN" ]; then
+    printf '%s:\n' "$f" >> "$TMPSKIP"
+    cat "$TMPUNKNOWN" >> "$TMPSKIP"
+    printf '(imported as custom)\n\n' >> "$TMPSKIP"
+  fi
+
+  # Build custom as JSON string with real newlines
+  if [ -s "$TMPUNKNOWN" ]; then
+    custom_json=$(jq -Rs '.' < "$TMPUNKNOWN")
+  else
+    custom_json=""
+  fi
+
+  # Build JSON from key=value lines
+  json=$(jq -Rn '
+    [inputs | select(contains("=")) |
+      . as $line | ($line | index("=")) as $i |
+      {($line[:$i]): ($line[$i+1:])}
+    ] | add
+  ' < "$TMPARGS")
+
+  # Add custom if present
+  if [ -n "$custom_json" ]; then
+    json=$(printf '%s' "$json" | jq --argjson r "$custom_json" '. + {custom: $r}')
+  fi
+
+  echo "Importing: secrets/$FOLDER/$entrykey"
+  printf '%s' "$json" | vault kv put "secrets/$FOLDER/$entrykey" -
+done
+
+if [ -s "$TMPSKIP" ]; then
+  echo ""
+  echo "=== NOT IMPORTED ==="
+  cat "$TMPSKIP"
+fi
+
+rm -f "$TMPARGS" "$TMPSKIP" "$TMPUNKNOWN" "$TMPBLOCKS"