发布于 

James添加收件钩子

前言

这个钩子是存储前的钩子,只能拿到MessageID拿不到uid信息

添加监听器

ReceiveEmailMailet,注意类文件路径,不是所有模块都可以添加监听器

当前类文件放在server/mailet/mailetcontainer-impl/src/main/java/org.apache.james.mailetcontainer/impl/matchers目录里面

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
/****************************************************************
* Licensed to the Apache Software Foundation (ASF) under one *
* or more contributor license agreements. See the NOTICE file *
* distributed with this work for additional information *
* regarding copyright ownership. The ASF licenses this file *
* to you under the Apache License, Version 2.0 (the *
* "License"); you may not use this file except in compliance *
* with the License. You may obtain a copy of the License at *
* *
* http://www.apache.org/licenses/LICENSE-2.0 *
* *
* Unless required by applicable law or agreed to in writing, *
* software distributed under the License is distributed on an *
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY *
* KIND, either express or implied. See the License for the *
* specific language governing permissions and limitations *
* under the License. *
****************************************************************/

package org.apache.james.mailetcontainer.impl.matchers;

import java.io.IOException;

import javax.mail.MessagingException;

import org.apache.mailet.Mail;
import org.apache.mailet.base.GenericMailet;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class ReceiveEmailMailet extends GenericMailet {

public static final Logger LOGGER = LoggerFactory.getLogger(ReceiveEmailMailet.class);

@Override
public void service(Mail mail) throws MessagingException {
LOGGER.info("收到邮件了{}",mail.getName());

LOGGER.info("收到邮件了,Subject: {}", mail.getMessage().getSubject());
LOGGER.info("收到邮件了,MessageID: {}", mail.getMessage().getMessageID());

try {
LOGGER.info("收到邮件了,Content: {}", mail.getMessage().getContent());
} catch (IOException e) {
LOGGER.error(e.toString());
}
}
}

修改配置文件

服务器conf目录下的mailetcontainer.xml文件

1
在<processor state="transport" enableJmx="true">标签中添加下面的代码

收发件都触发

1
<mailet match="All" class="org.apache.james.mailetcontainer.impl.matchers.ReceiveEmailMailet" />

只在收件触发

1
<mailet match="RecipientIsLocal" class="org.apache.james.mailetcontainer.impl.matchers.ReceiveEmailMailet" />

重启James

1
./bin/james-server.sh stop
1
./bin/james-server.sh start

或者jps找到james的进程再kill -9 杀掉进程

1
2
3
4
5
6
7
8
9
10
root@touchs-test:/data/apps/mail# jps
1092272 touchs.jar
1110035 Jps
24307 agent-2.11.11.jar
682496 james-server-distributed-app.jar
72491 OpenSearch
72492 PerformanceAnalyzerApp
240781 server-2.11.11.jar
root@touchs-test:/data/apps/mail# kill -9 682496
root@touchs-test:/data/apps/mail#
1
java -Dworking.directory=. -Dlogback.configurationFile=conf/logback.xml -Djdk.tls.ephemeralDHKeySize=2048 -jar james-server-distributed-app.jar &

拓展

触发后调用异步调用接口

接收端

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
import com.touchsmail.common.domain.AjaxResult;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@Slf4j
@RestController
@RequestMapping("/api/mail/hook")
public class EmailHookController {

/**
* 邮箱服务器接收邮件回调
*/
@ApiOperation(value = "邮箱服务器接收邮件回调")
@PostMapping("/receive")
public AjaxResult receiveEmails(@RequestBody String param) {
return AjaxResult.success(param);
}

}

推送端

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
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
/****************************************************************
* Licensed to the Apache Software Foundation (ASF) under one *
* or more contributor license agreements. See the NOTICE file *
* distributed with this work for additional information *
* regarding copyright ownership. The ASF licenses this file *
* to you under the Apache License, Version 2.0 (the *
* "License"); you may not use this file except in compliance *
* with the License. You may obtain a copy of the License at *
* *
* http://www.apache.org/licenses/LICENSE-2.0 *
* *
* Unless required by applicable law or agreed to in writing, *
* software distributed under the License is distributed on an *
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY *
* KIND, either express or implied. See the License for the *
* specific language governing permissions and limitations *
* under the License. *
****************************************************************/

package org.apache.james.mailetcontainer.impl.matchers;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.util.Arrays;
import java.util.Base64;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CompletableFuture;
import java.util.stream.Collectors;

import javax.mail.BodyPart;
import javax.mail.Message;
import javax.mail.MessagingException;
import javax.mail.Multipart;
import javax.mail.internet.InternetAddress;
import javax.mail.internet.MimeMessage;

import org.apache.commons.lang3.StringUtils;
import org.apache.mailet.Mail;
import org.apache.mailet.base.GenericMailet;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.fasterxml.jackson.databind.ObjectMapper;


public class ReceiveEmailMailet extends GenericMailet {

public static final Logger LOGGER = LoggerFactory.getLogger(ReceiveEmailMailet.class);

@Override
public void service(Mail mail) throws MessagingException {

LOGGER.info("收件钩子:Subject: {}", mail.getMessage().getSubject());
LOGGER.info("收件钩子:MessageID: {}", mail.getMessage().getMessageID());

try {
sendAsyncHttpRequest(mail);
} catch (IOException e) {
LOGGER.error(e.toString());
}
}

private void sendAsyncHttpRequest(Mail mail) throws MessagingException, IOException {
HttpClient httpClient = HttpClient.newHttpClient();
Map<String, Object> stringObjectMap = buildRequestBody(mail);
ObjectMapper objectMapper = new ObjectMapper();
String requestBody = objectMapper.writeValueAsString(stringObjectMap);
HttpRequest request = HttpRequest.newBuilder()
.uri(URI.create("http://127.0.0.1:6001/api/mail/hook/receive"))
.header("Content-Type", "application/json")
.POST(HttpRequest.BodyPublishers.ofString(requestBody))
.build();
// 发送异步请求
CompletableFuture<HttpResponse<String>> futureResponse = httpClient.sendAsync(request, HttpResponse.BodyHandlers.ofString());
// 处理异步响应
futureResponse.thenAccept(response -> {
int statusCode = response.statusCode();
String responseBody = response.body();

if (statusCode == 200) {
LOGGER.info("收件钩子:推送成功");
} else {
LOGGER.error("收件钩子:邮件推送失败,状态码: {} ,响应内容: {} ", statusCode, responseBody);
}

}).exceptionally(ex -> {
LOGGER.error("收件钩子:邮件推送时发生异常: {}", ex.toString());
return null;
});
}

private Map<String, Object> buildRequestBody(Mail mail) throws MessagingException, IOException {
Map<String, Object> emailData = new HashMap<>();
MimeMessage message = mail.getMessage();
// 异步去获取邮件正文
CompletableFuture<Object> futureResult = handleRelatedContentAsync(message);
List<String> to = Arrays.stream(message.getRecipients(Message.RecipientType.TO)).map(m -> ((InternetAddress) m).getAddress()).collect(Collectors.toList());
List<String> from = Arrays.stream(message.getFrom()).map(m -> ((InternetAddress) m).getAddress()).collect(Collectors.toList());

emailData.put("subject", message.getSubject());
emailData.put("from", from);
emailData.put("to", to);
emailData.put("messageID", mail.getMessage().getMessageID());

emailData.put("date", message.getReceivedDate());
emailData.put("content", futureResult.join());
return emailData;
}


private CompletableFuture<Object> handleRelatedContentAsync(Message message) {
return CompletableFuture.supplyAsync(() -> {
try {
if (message.isMimeType("text/plain") || message.isMimeType("text/html")) {
return message.getContent().toString();
} else if (message.isMimeType("multipart/*")) {
Multipart multipart = (Multipart) message.getContent();
StringBuilder htmlContent = new StringBuilder();
Map<String, String> imageMap = new HashMap<>();
buildMultipartResout(multipart, htmlContent, imageMap);
return htmlContent.toString();
}
return "unknown content format";
} catch (Exception e) {
// 记录错误日志
LOGGER.error("Error processing related content", e);
return "error";
}
});
}

private void buildMultipartResout(Multipart multipart, StringBuilder htmlContent, Map<String, String> imageMap) throws Exception {

// 遍历所有部分,收集HTML内容和图片
for (int i = 0; i < multipart.getCount(); i++) {
BodyPart part = multipart.getBodyPart(i);
// 处理 multipart/related 类型
if (part.getContentType().contains("multipart/related") || part.getContentType().contains("multipart/RELATED")) {
buildMultipartResout((Multipart) part.getContent(), htmlContent, imageMap);
} else if (part.isMimeType("multipart/alternative") || part.getContentType().contains("multipart/ALTERNATIVE")) {
processMultipartAlternative(part, htmlContent, imageMap);
} else if (part.isMimeType("image/*")) {
processImg(part, imageMap);
} else if (part.getContent() instanceof Multipart) {
processMultipartAlternative(part, htmlContent, imageMap);
} else if (part.isMimeType("text/html")) {
htmlContent.append(part.getContent().toString());
} else {
// 可能是附件
LOGGER.warn("其他部分 ContentType: " + part.getContentType());
}
}

String content = getContent(htmlContent, imageMap);
htmlContent.setLength(0);
htmlContent.append(StringUtils.isEmpty(content) ? "unknown content format" : content);
}

private void processMultipartAlternative(BodyPart part, StringBuilder htmlContent, Map<String, String> imageMap) throws Exception {
if (part.getContent() instanceof Multipart) {
Multipart alternative = (Multipart) part.getContent();

for (int i = 0; i < alternative.getCount(); i++) {
BodyPart subPart = alternative.getBodyPart(i);

if (subPart.isMimeType("text/html")) {
htmlContent.append(subPart.getContent()).append("<br/>");
} else if (subPart.isMimeType("image/*")) {
processImg(subPart, imageMap);
} else if (subPart.isMimeType("multipart/alternative") || subPart.getContentType().contains("multipart/ALTERNATIVE")) {
processMultipartAlternative(subPart, htmlContent, imageMap);
} else if (subPart.getContentType().contains("multipart/related") || subPart.getContentType().contains("multipart/RELATED")) {
processMultipartAlternative(subPart, htmlContent, imageMap);
} else {
LOGGER.warn("[processMultipartAlternative]其他部分 ContentType: " + subPart.getContentType());
}
}
} else {
LOGGER.warn("multipart/ALTERNATIVE 的内容不是 Multipart 类型");
}
}

private static String getContent(StringBuilder htmlContent, Map<String, String> imageMap) {
String content = htmlContent.toString();

// 如果找到HTML内容和图片,进行替换
if (!imageMap.isEmpty()) {
for (Map.Entry<String, String> entry : imageMap.entrySet()) {
String contentId = entry.getKey();
String base64Image = entry.getValue();

// 替换HTML中的图片引用
String imgPattern = "cid:" + contentId;
String base64Pattern = "data:image/jpeg;base64," + base64Image;
content = content.replace(imgPattern, base64Pattern);
}
}
return content;
}

private void processImg(BodyPart part, Map<String, String> imageMap) throws MessagingException, IOException {
// 获取Content-ID
String[] contentIds = part.getHeader("Content-ID");
if (contentIds != null && contentIds.length > 0) {
String contentId = contentIds[0].replaceAll("[<>]", "");
// 转换图片为base64
String base64Image = convertImageToBase64(part);
imageMap.put(contentId, base64Image);
}
}

private String convertImageToBase64(BodyPart part) throws MessagingException, IOException {
try (InputStream is = part.getInputStream()) {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
byte[] buffer = new byte[4096];
int bytesRead;
while ((bytesRead = is.read(buffer)) != -1) {
baos.write(buffer, 0, bytesRead);
}
return Base64.getEncoder().encodeToString(baos.toByteArray());
}
}
}

编译

依赖注意顺序,否则编译会报错,不知道顺序可以问一下ai

在该模块的根目录下编译,编译好之后把jar包放到服务器james的依赖目录(james-server-distributed-app.lib)里面,然后重启James

1
2
3
edy@EDYs-MacBook-Air mailetcontainer-impl % pwd
/Users/edy/Documents/ProjectCode/James382/JamesByKre/server/mailet/mailetcontainer-impl
edy@EDYs-MacBook-Air mailetcontainer-impl % mvn clean install -Dmaven.test.skip=true -e
1
mvn clean install -Dmaven.test.skip=true -e