-
Notifications
You must be signed in to change notification settings - Fork 1
/
maildir.go
95 lines (79 loc) · 2.28 KB
/
maildir.go
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
package maildir
import (
"errors"
"os"
"path/filepath"
"sort"
"strings"
"github.com/amalfra/maildir/v3/lib"
)
// Maildir implements maildir format and it's operations
type Maildir struct {
path string
}
// NewMaildir will create new maildir at specified path
func NewMaildir(path string) *Maildir {
maildir := new(Maildir)
maildir.path = path
_, dCur := os.Stat(path + "/cur")
_, dTmp := os.Stat(path + "/tmp")
_, dNew := os.Stat(path + "/new")
if os.IsNotExist(dCur) || os.IsNotExist(dTmp) || os.IsNotExist(dNew) {
maildir.createDirectories()
}
return maildir
}
// createDirectories will the sub directories required by maildir
func (m *Maildir) createDirectories() {
for _, subDir := range lib.Subdirs {
os.MkdirAll(filepath.Join(m.path, subDir), os.ModePerm)
}
}
// Add writes data out as a new message. Returns Message instance
func (m *Maildir) Add(data string) (*lib.Message, error) {
msg, err := lib.NewMessage(m.path)
if err != nil {
return nil, errors.New("failed to create message")
}
err = msg.Write(data)
if err != nil {
return nil, errors.New("failed to write message")
}
return msg, nil
}
// Get returns a message object for key
func (m *Maildir) Get(key string) *lib.Message {
return lib.LoadMessage(m.path, key)
}
// List returns an array of messages from new or cur directory, sorted by key
func (m *Maildir) List(dir string) (map[string]*lib.Message, error) {
if !lib.StringInSlice(dir, lib.Subdirs) {
return nil, errors.New("dir must be :new, :cur, or :tmp")
}
keys, err := m.getDirListing(dir)
if err != nil {
return nil, errors.New("failed to get directory listing")
}
sort.Sort(sort.StringSlice(keys))
// map keys to message objects
keyMap := make(map[string]*lib.Message)
for _, key := range keys {
keyMap[key] = m.Get(key)
}
return keyMap, nil
}
// getDirListing returns an array of keys in dir
func (m *Maildir) getDirListing(dir string) ([]string, error) {
filter := "*"
searchPath := filepath.Join(m.path, dir, filter)
filePaths, err := filepath.Glob(searchPath)
// remove maildir path so that only key remains
for i, filePath := range filePaths {
filePaths[i] = strings.TrimPrefix(filePath, m.path)
}
return filePaths, err
}
// Delete a message by key
func (m *Maildir) Delete(key string) error {
return m.Get(key).Destroy()
}