keygen.go 1.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778
  1. package main
  2. import (
  3. "flag"
  4. "fmt"
  5. "io"
  6. "os"
  7. "github.com/slackhq/nebula/cert"
  8. )
  9. type keygenFlags struct {
  10. set *flag.FlagSet
  11. outKeyPath *string
  12. outPubPath *string
  13. curve *string
  14. }
  15. func newKeygenFlags() *keygenFlags {
  16. cf := keygenFlags{set: flag.NewFlagSet("keygen", flag.ContinueOnError)}
  17. cf.set.Usage = func() {}
  18. cf.outPubPath = cf.set.String("out-pub", "", "Required: path to write the public key to")
  19. cf.outKeyPath = cf.set.String("out-key", "", "Required: path to write the private key to")
  20. cf.curve = cf.set.String("curve", "25519", "ECDH Curve (25519, P256)")
  21. return &cf
  22. }
  23. func keygen(args []string, out io.Writer, errOut io.Writer) error {
  24. cf := newKeygenFlags()
  25. err := cf.set.Parse(args)
  26. if err != nil {
  27. return err
  28. }
  29. if err := mustFlagString("out-key", cf.outKeyPath); err != nil {
  30. return err
  31. }
  32. if err := mustFlagString("out-pub", cf.outPubPath); err != nil {
  33. return err
  34. }
  35. var pub, rawPriv []byte
  36. var curve cert.Curve
  37. switch *cf.curve {
  38. case "25519", "X25519", "Curve25519", "CURVE25519":
  39. pub, rawPriv = x25519Keypair()
  40. curve = cert.Curve_CURVE25519
  41. case "P256":
  42. pub, rawPriv = p256Keypair()
  43. curve = cert.Curve_P256
  44. default:
  45. return fmt.Errorf("invalid curve: %s", *cf.curve)
  46. }
  47. err = os.WriteFile(*cf.outKeyPath, cert.MarshalPrivateKey(curve, rawPriv), 0600)
  48. if err != nil {
  49. return fmt.Errorf("error while writing out-key: %s", err)
  50. }
  51. err = os.WriteFile(*cf.outPubPath, cert.MarshalPublicKey(curve, pub), 0600)
  52. if err != nil {
  53. return fmt.Errorf("error while writing out-pub: %s", err)
  54. }
  55. return nil
  56. }
  57. func keygenSummary() string {
  58. return "keygen <flags>: create a public/private key pair. the public key can be passed to `nebula-cert sign`"
  59. }
  60. func keygenHelp(out io.Writer) {
  61. cf := newKeygenFlags()
  62. out.Write([]byte("Usage of " + os.Args[0] + " " + keygenSummary() + "\n"))
  63. cf.set.SetOutput(out)
  64. cf.set.PrintDefaults()
  65. }