File: responses.go

package info (click to toggle)
golang-github-emersion-go-imap-sortthread 1.2.0-2
  • links: PTS, VCS
  • area: main
  • in suites: bookworm, forky, sid, trixie
  • size: 116 kB
  • sloc: makefile: 2
file content (120 lines) | stat: -rw-r--r-- 2,739 bytes parent folder | download
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
package sortthread

import (
	"strconv"

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

type SortResponse struct {
	Ids []uint32
}

type ThreadResponse struct {
	Threads []*Thread
}

func (r *SortResponse) Handle(resp imap.Resp) error {
	name, fields, ok := imap.ParseNamedResp(resp)
	if !ok || name != "SORT" {
		return responses.ErrUnhandled
	}

	r.Ids = make([]uint32, len(fields))
	for i, f := range fields {
		if id, err := imap.ParseNumber(f); err != nil {
			return err
		} else {
			r.Ids[i] = id
		}
	}

	return nil
}

func (r *SortResponse) WriteTo(w *imap.Writer) error {
	fields := make([]interface{}, 0, len(r.Ids)+1)
	fields = append(fields, imap.RawString("SORT"))
	for _, id := range r.Ids {
		fields = append(fields, imap.RawString(strconv.FormatInt(int64(id), 10)))
	}

	return imap.NewUntaggedResp(fields).WriteTo(w)
}

func (r *ThreadResponse) Handle(resp imap.Resp) error {
	name, fields, ok := imap.ParseNamedResp(resp)
	if !ok || name != "THREAD" {
		return responses.ErrUnhandled
	}
	if threads, err := parseThreadResp(fields); err != nil {
		return err
	} else {
		r.Threads = threads
	}

	return nil
}

func parseThreadResp(fields []interface{}) ([]*Thread, error) {
	var parent *Thread
	var siblings []*Thread
	for _, f := range fields {
		switch f := f.(type) {
		case string, imap.RawString:
			id, err := imap.ParseNumber(f)
			if err != nil {
				return nil, err
			}
			t := Thread{Id: id}
			if parent == nil {
				siblings = append(siblings, &t)
			} else {
				parent.Children = append(t.Children, &t)
			}
			parent = &t
		case []interface{}:
			t, err := parseThreadResp(f)
			if err != nil {
				return nil, err
			}
			// Parent doesn't exist, e.g. didn't match the search
			// criteria. Let's ignore the parent thread.
			if parent == nil {
				siblings = append(siblings, t...)
			} else {
				parent.Children = append(parent.Children, t...)
			}
		default:
			return nil, responses.ErrUnhandled
		}
	}
	return siblings, nil
}

func formatThread(thread *Thread) []interface{} {
	f := make([]interface{}, 0, 1+len(thread.Children))
	f = append(f, imap.RawString(strconv.FormatInt(int64(thread.Id), 10)))
	if len(thread.Children) == 1 {
		f = append(f, formatThread(thread.Children[0])...)
	} else {
		for _, c := range thread.Children {
			f = append(f, formatThread(c))
		}
	}
	return f
}

func formatThreadResp(threads []*Thread) []interface{} {
	fields := make([]interface{}, 0, len(threads)+1)
	fields = append(fields, imap.RawString("THREAD"))
	for _, t := range threads {
		fields = append(fields, formatThread(t))
	}
	return fields
}

func (r *ThreadResponse) WriteTo(w *imap.Writer) error {
	return imap.NewUntaggedResp(formatThreadResp(r.Threads)).WriteTo(w)
}