hydroxide-push/protonmail/messages.go

248 lines
5.1 KiB
Go

package protonmail
import (
"bytes"
"io"
"net/http"
"strings"
"golang.org/x/crypto/openpgp"
"golang.org/x/crypto/openpgp/armor"
)
type MessageType int
const (
MessageInbox MessageType = iota
MessageDraft
MessageSent
MessageInboxAndSent
)
type MessageEncryption int
const (
MessageUnencrypted MessageEncryption = iota
MessageEncryptedInternal
MessageEncryptedExternal
MessageEncryptedOutside
_
_
_
MessageEncryptedInlinePGP
MessageEncryptedPGPMIME
_
_
)
type MessageAddress struct {
Address string
Name string
}
type Message struct {
ID string
Order int64
Subject string
IsRead int
Type MessageType
Sender *MessageAddress
ReplyTo *MessageAddress
ToList []*MessageAddress
Time int64
Size int64
IsEncrypted MessageEncryption
ExpirationTime int64
IsReplied int
IsRepliedAll int
IsForwarded int
SpamScore int
AddressID string
Body string
MIMEType string
CCList []*MessageAddress
BCCList []*MessageAddress
Header string
Attachments []*Attachment
LabelIDs []string
ExternalID string
}
func (msg *Message) Read(keyring openpgp.KeyRing, prompt openpgp.PromptFunction) (*openpgp.MessageDetails, error) {
switch msg.IsEncrypted {
case MessageUnencrypted:
return &openpgp.MessageDetails{
IsEncrypted: false,
IsSigned: false,
UnverifiedBody: strings.NewReader(msg.Body),
}, nil
default:
block, err := armor.Decode(strings.NewReader(msg.Body))
if err != nil {
return nil, err
}
return openpgp.ReadMessage(block.Body, keyring, prompt, nil)
}
}
type messageWriter struct {
plaintext io.WriteCloser
ciphertext io.WriteCloser
b *bytes.Buffer
msg *Message
}
func (w *messageWriter) Write(p []byte) (n int, err error) {
return w.plaintext.Write(p)
}
func (w *messageWriter) Close() error {
if err := w.plaintext.Close(); err != nil {
return err
}
if err := w.ciphertext.Close(); err != nil {
return err
}
w.msg.Body = w.b.String()
return nil
}
func (msg *Message) Encrypt(to []*openpgp.Entity, signed *openpgp.Entity) (plaintext io.WriteCloser, err error) {
var b bytes.Buffer
ciphertext, err := armor.Encode(&b, "PGP MESSAGE", nil)
if err != nil {
return nil, err
}
plaintext, err = openpgp.Encrypt(ciphertext, to, signed, nil, nil)
if err != nil {
return nil, err
}
return &messageWriter{
plaintext: plaintext,
ciphertext: ciphertext,
b: &b,
msg: msg,
}, nil
}
func (c *Client) GetMessage(id string) (*Message, error) {
req, err := c.newRequest(http.MethodGet, "/messages/"+id, nil)
if err != nil {
return nil, err
}
var respData struct {
resp
Message *Message
}
if err := c.doJSON(req, &respData); err != nil {
return nil, err
}
return respData.Message, nil
}
// CreateDraftMessage creates a new draft message. ToList, CCList, BCCList,
// Subject, Body and AddressID are required in msg.
func (c *Client) CreateDraftMessage(msg *Message, parentID string) (*Message, error) {
reqData := struct {
Message *Message
ParentID string `json:",omitempty"`
Action *int `json:",omitempty"`
}{msg, parentID, nil}
req, err := c.newJSONRequest(http.MethodPost, "/messages", &reqData)
if err != nil {
return nil, err
}
var respData struct {
resp
Message *Message
}
if err := c.doJSON(req, &respData); err != nil {
return nil, err
}
return respData.Message, nil
}
func (c *Client) UpdateDraftMessage(msg *Message) (*Message, error) {
reqData := struct {
Message *Message
}{msg}
req, err := c.newJSONRequest(http.MethodPut, "/messages/"+msg.ID, &reqData)
if err != nil {
return nil, err
}
var respData struct {
resp
Message *Message
}
if err := c.doJSON(req, &respData); err != nil {
return nil, err
}
return respData.Message, nil
}
type MessageKeyPacket struct {
ID string
KeyPackets string
}
type MessagePackageType int
const (
MessagePackageInternal MessagePackageType = 1
MessagePackageEncryptedOutside = 2
MessagePackageClear = 4
MessagePackageInlinePGP = 8
MessagePackagePGPMIME = 16
MessagePackageMIME = 32
)
type MessagePackage struct {
Address string
Type MessagePackageType
Body string
KeyPackets []*MessageKeyPacket
Token string
EncToken string
PasswordHint string `json:",omitempty"`
}
type MessageOutgoing struct {
ID string
// Only if there's a recipient without a public key
ClearBody string
AttachmentKeys []*AttachmentKey
// Only if message expires
ExpirationTime int // duration in seconds
Packages []*MessagePackage
}
func (c *Client) SendMessage(msg *MessageOutgoing) (sent, parent *Message, err error) {
req, err := c.newJSONRequest(http.MethodPut, "/messages/"+msg.ID, msg)
if err != nil {
return nil, nil, err
}
var respData struct {
resp
Sent, Parent *Message
}
if err := c.doJSON(req, &respData); err != nil {
return nil, nil, err
}
return respData.Sent, respData.Parent, nil
}