aboutsummaryrefslogtreecommitdiff
path: root/vendor/github.com/emersion/go-imap/responses/select.go
blob: e450963b537e0c014d273a0240bf02e9a0443a8e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
package responses

import (
	"fmt"

	"github.com/emersion/go-imap"
)

// A SELECT response.
type Select struct {
	Mailbox *imap.MailboxStatus
}

func (r *Select) Handle(resp imap.Resp) error {
	if r.Mailbox == nil {
		r.Mailbox = &imap.MailboxStatus{Items: make(map[imap.StatusItem]interface{})}
	}
	mbox := r.Mailbox

	switch resp := resp.(type) {
	case *imap.DataResp:
		name, fields, ok := imap.ParseNamedResp(resp)
		if !ok || name != "FLAGS" {
			return ErrUnhandled
		} else if len(fields) < 1 {
			return errNotEnoughFields
		}

		flags, _ := fields[0].([]interface{})
		mbox.Flags, _ = imap.ParseStringList(flags)
	case *imap.StatusResp:
		if len(resp.Arguments) < 1 {
			return ErrUnhandled
		}

		var item imap.StatusItem
		switch resp.Code {
		case "UNSEEN":
			mbox.UnseenSeqNum, _ = imap.ParseNumber(resp.Arguments[0])
		case "PERMANENTFLAGS":
			flags, _ := resp.Arguments[0].([]interface{})
			mbox.PermanentFlags, _ = imap.ParseStringList(flags)
		case "UIDNEXT":
			mbox.UidNext, _ = imap.ParseNumber(resp.Arguments[0])
			item = imap.StatusUidNext
		case "UIDVALIDITY":
			mbox.UidValidity, _ = imap.ParseNumber(resp.Arguments[0])
			item = imap.StatusUidValidity
		default:
			return ErrUnhandled
		}

		if item != "" {
			mbox.ItemsLocker.Lock()
			mbox.Items[item] = nil
			mbox.ItemsLocker.Unlock()
		}
	default:
		return ErrUnhandled
	}
	return nil
}

func (r *Select) WriteTo(w *imap.Writer) error {
	mbox := r.Mailbox

	if mbox.Flags != nil {
		flags := make([]interface{}, len(mbox.Flags))
		for i, f := range mbox.Flags {
			flags[i] = imap.RawString(f)
		}
		res := imap.NewUntaggedResp([]interface{}{imap.RawString("FLAGS"), flags})
		if err := res.WriteTo(w); err != nil {
			return err
		}
	}

	if mbox.PermanentFlags != nil {
		flags := make([]interface{}, len(mbox.PermanentFlags))
		for i, f := range mbox.PermanentFlags {
			flags[i] = imap.RawString(f)
		}
		statusRes := &imap.StatusResp{
			Type:      imap.StatusRespOk,
			Code:      imap.CodePermanentFlags,
			Arguments: []interface{}{flags},
			Info:      "Flags permitted.",
		}
		if err := statusRes.WriteTo(w); err != nil {
			return err
		}
	}

	if mbox.UnseenSeqNum > 0 {
		statusRes := &imap.StatusResp{
			Type:      imap.StatusRespOk,
			Code:      imap.CodeUnseen,
			Arguments: []interface{}{mbox.UnseenSeqNum},
			Info:      fmt.Sprintf("Message %d is first unseen", mbox.UnseenSeqNum),
		}
		if err := statusRes.WriteTo(w); err != nil {
			return err
		}
	}

	for k := range r.Mailbox.Items {
		switch k {
		case imap.StatusMessages:
			res := imap.NewUntaggedResp([]interface{}{mbox.Messages, imap.RawString("EXISTS")})
			if err := res.WriteTo(w); err != nil {
				return err
			}
		case imap.StatusRecent:
			res := imap.NewUntaggedResp([]interface{}{mbox.Recent, imap.RawString("RECENT")})
			if err := res.WriteTo(w); err != nil {
				return err
			}
		case imap.StatusUidNext:
			statusRes := &imap.StatusResp{
				Type:      imap.StatusRespOk,
				Code:      imap.CodeUidNext,
				Arguments: []interface{}{mbox.UidNext},
				Info:      "Predicted next UID",
			}
			if err := statusRes.WriteTo(w); err != nil {
				return err
			}
		case imap.StatusUidValidity:
			statusRes := &imap.StatusResp{
				Type:      imap.StatusRespOk,
				Code:      imap.CodeUidValidity,
				Arguments: []interface{}{mbox.UidValidity},
				Info:      "UIDs valid",
			}
			if err := statusRes.WriteTo(w); err != nil {
				return err
			}
		}
	}

	return nil
}