/
filter-prepend.go
111 lines (96 loc) · 2.81 KB
/
filter-prepend.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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
package main
import (
"bufio"
"flag"
"fmt"
"log"
"mime"
"os"
"strings"
)
var dec *mime.WordDecoder
var prefix string
var encPrefix string
var forceEncode bool
var sessionIndex int
var opaqueIndex int
var extraPrefix string
var encExtraPrefix string
func init() {
flag.StringVar(&prefix, "prefix", "[*EXT*]", "Prepend subject with <prefix> if not already present")
flag.BoolVar(&forceEncode, "encode", false, "Encode prefix whether subject is encoded or not")
flag.StringVar(&extraPrefix, "extraprefix", "[EXT]", "Also detect <extraprefix> within the subject")
flag.Parse()
}
func main() {
scanner := bufio.NewScanner(os.Stdin)
dec = new(mime.WordDecoder)
encPrefix = mime.QEncoding.Encode("utf-8", prefix)
encExtraPrefix = mime.QEncoding.Encode("utf-8", extraPrefix)
for scanner.Scan() {
line := scanner.Text()
if strings.HasPrefix(line, "config|smtpd-version") {
SetIndexValue(line)
} else if strings.HasPrefix(line, "config|ready") {
RegisterFilter()
log.Println("filter-prepend registered with " + prefix + " and extraPrefix " + extraPrefix)
if forceEncode {
log.Println("filter-prepend will always encode prefix to " + encPrefix)
}
} else {
dataSplit := strings.Split(line, "|")
if len(dataSplit) >= 8 {
if dataSplit[4] == "data-line" {
DoDataLine(dataSplit)
}
}
}
}
if err := scanner.Err(); err != nil {
log.Println(err)
}
}
func SetIndexValue(line string) {
version := strings.Split(line, "|")[2]
if strings.HasPrefix(version, "6.6.") {
sessionIndex = 6
opaqueIndex = 5
log.Println("smtpd 6.6 detected : swapping session and opaque indices.")
} else {
sessionIndex = 5
opaqueIndex = 6
}
}
func RegisterFilter() {
fmt.Println("register|filter|smtp-in|data-line")
fmt.Println("register|report|smtp-in|link-disconnect")
fmt.Println("register|report|smtp-in|link-connect")
fmt.Println("register|ready")
}
func DoDataLine(dataSplit []string) {
if strings.HasPrefix(strings.ToUpper(dataSplit[7]), "SUBJECT: ") {
fmt.Printf("filter-dataline|%s|%s|%s\n", dataSplit[sessionIndex], dataSplit[opaqueIndex], ProcessSubject(dataSplit[7:]))
} else {
fmt.Printf("filter-dataline|%s|%s|%s\n", dataSplit[sessionIndex], dataSplit[opaqueIndex], strings.Join(dataSplit[7:], "|"))
}
}
func ProcessSubject(s []string) string {
result := ""
subject := ""
isEncoded := false
rawsub := strings.SplitAfterN(strings.Join(s, "|"), ": ", 2)[1]
if len(rawsub) < 8 || !strings.HasPrefix(rawsub, "=?") || !strings.HasSuffix(rawsub, "?=") || strings.Count(rawsub, "?") != 4 {
subject = rawsub
} else {
isEncoded = true
subject, _ = dec.Decode(rawsub)
}
if !strings.Contains(subject, prefix) && !strings.Contains(subject, extraPrefix) {
if isEncoded || forceEncode {
result = encPrefix + " "
} else {
result = prefix + " "
}
}
return "Subject: " + result + rawsub
}