2019-06-03 17:00:01 +02:00
|
|
|
package helper
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"strings"
|
|
|
|
|
|
2019-12-27 19:35:43 +01:00
|
|
|
"github.com/ProtonMail/gopenpgp/v2/crypto"
|
|
|
|
|
"github.com/ProtonMail/gopenpgp/v2/internal"
|
2019-06-03 17:00:01 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// SignCleartextMessageArmored signs text given a private key and its passphrase, canonicalizes and trims the newlines,
|
|
|
|
|
// and returns the PGP-compliant special armoring
|
2019-12-27 19:35:43 +01:00
|
|
|
func SignCleartextMessageArmored(privateKey string, passphrase []byte, text string) (string, error) {
|
|
|
|
|
signingKey, err := crypto.NewKeyFromArmored(privateKey)
|
2019-06-03 17:00:01 +02:00
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-27 19:35:43 +01:00
|
|
|
unlockedKey, err := signingKey.Unlock(passphrase)
|
2019-06-03 17:00:01 +02:00
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
2020-04-27 21:01:23 +02:00
|
|
|
defer unlockedKey.ClearPrivateParams()
|
2019-06-03 17:00:01 +02:00
|
|
|
|
2019-12-27 19:35:43 +01:00
|
|
|
keyRing, err := crypto.NewKeyRing(unlockedKey)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return SignCleartextMessage(keyRing, text)
|
2019-06-03 17:00:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// VerifyCleartextMessageArmored verifies PGP-compliant armored signed plain text given the public key
|
|
|
|
|
// and returns the text or err if the verification fails
|
|
|
|
|
func VerifyCleartextMessageArmored(publicKey, armored string, verifyTime int64) (string, error) {
|
2019-12-27 19:35:43 +01:00
|
|
|
signingKey, err := crypto.NewKeyFromArmored(publicKey)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
verifyKeyRing, err := crypto.NewKeyRing(signingKey)
|
2019-06-03 17:00:01 +02:00
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return VerifyCleartextMessage(verifyKeyRing, armored, verifyTime)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// SignCleartextMessage signs text given a private keyring, canonicalizes and trims the newlines,
|
|
|
|
|
// and returns the PGP-compliant special armoring
|
|
|
|
|
func SignCleartextMessage(keyRing *crypto.KeyRing, text string) (string, error) {
|
|
|
|
|
text = canonicalizeAndTrim(text)
|
|
|
|
|
message := crypto.NewPlainMessageFromString(text)
|
|
|
|
|
|
|
|
|
|
signature, err := keyRing.SignDetached(message)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-02 07:36:02 -07:00
|
|
|
return crypto.NewClearTextMessage(message.GetBinary(), signature.GetBinary()).GetArmored()
|
2019-06-03 17:00:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// VerifyCleartextMessage verifies PGP-compliant armored signed plain text given the public keyring
|
|
|
|
|
// and returns the text or err if the verification fails
|
|
|
|
|
func VerifyCleartextMessage(keyRing *crypto.KeyRing, armored string, verifyTime int64) (string, error) {
|
2019-07-02 07:36:02 -07:00
|
|
|
clearTextMessage, err := crypto.NewClearTextMessageFromArmored(armored)
|
2019-06-03 17:00:01 +02:00
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-02 07:36:02 -07:00
|
|
|
message := crypto.NewPlainMessageFromString(clearTextMessage.GetString())
|
2019-08-19 12:06:30 +02:00
|
|
|
signature := crypto.NewPGPSignature(clearTextMessage.GetBinarySignature())
|
2019-07-02 07:36:02 -07:00
|
|
|
err = keyRing.VerifyDetached(message, signature, verifyTime)
|
2019-06-03 17:00:01 +02:00
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return message.GetString(), nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ----- INTERNAL FUNCTIONS -----
|
|
|
|
|
|
|
|
|
|
// canonicalizeAndTrim alters a string canonicalizing and trimming the newlines
|
|
|
|
|
func canonicalizeAndTrim(text string) string {
|
|
|
|
|
text = internal.TrimNewlines(text)
|
|
|
|
|
text = strings.Replace(strings.Replace(text, "\r\n", "\n", -1), "\n", "\r\n", -1)
|
|
|
|
|
return text
|
|
|
|
|
}
|