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
|
/*
* Copyright 2018 - 2020 Anton Tananaev (anton@traccar.org)
* Copyright 2018 Andrey Kunitsyn (andrey@traccar.org)
*
* Licensed 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.traccar.sms;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.traccar.Context;
import org.traccar.config.Keys;
import org.traccar.helper.DataConverter;
import org.traccar.notification.MessageException;
import javax.ws.rs.client.Entity;
import javax.ws.rs.client.Invocation;
import javax.ws.rs.client.InvocationCallback;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
public class HttpSmsClient implements SmsManager {
private static final Logger LOGGER = LoggerFactory.getLogger(HttpSmsClient.class);
private final String url;
private final String authorizationHeader;
private final String authorization;
private final String template;
private final boolean encode;
private final MediaType mediaType;
public HttpSmsClient() {
url = Context.getConfig().getString(Keys.SMS_HTTP_URL);
authorizationHeader = Context.getConfig().getString(Keys.SMS_HTTP_AUTHORIZATION_HEADER);
if (Context.getConfig().hasKey(Keys.SMS_HTTP_AUTHORIZATION)) {
authorization = Context.getConfig().getString(Keys.SMS_HTTP_AUTHORIZATION);
} else {
String user = Context.getConfig().getString(Keys.SMS_HTTP_USER);
String password = Context.getConfig().getString(Keys.SMS_HTTP_PASSWORD);
if (user != null && password != null) {
authorization = "Basic "
+ DataConverter.printBase64((user + ":" + password).getBytes(StandardCharsets.UTF_8));
} else {
authorization = null;
}
}
template = Context.getConfig().getString(Keys.SMS_HTTP_TEMPLATE).trim();
if (template.charAt(0) == '{' || template.charAt(0) == '[') {
encode = false;
mediaType = MediaType.APPLICATION_JSON_TYPE;
} else {
encode = true;
mediaType = MediaType.APPLICATION_FORM_URLENCODED_TYPE;
}
}
private String prepareValue(String value) throws UnsupportedEncodingException {
return encode ? URLEncoder.encode(value, StandardCharsets.UTF_8.name()) : value;
}
private String preparePayload(String destAddress, String message) {
try {
return template
.replace("{phone}", prepareValue(destAddress))
.replace("{message}", prepareValue(message.trim()));
} catch (UnsupportedEncodingException e) {
throw new RuntimeException(e);
}
}
private Invocation.Builder getRequestBuilder() {
Invocation.Builder builder = Context.getClient().target(url).request();
if (authorization != null) {
builder = builder.header(authorizationHeader, authorization);
}
return builder;
}
@Override
public void sendMessageSync(String destAddress, String message, boolean command) throws MessageException {
Response response = getRequestBuilder().post(Entity.entity(preparePayload(destAddress, message), mediaType));
if (response.getStatus() / 100 != 2) {
throw new MessageException(response.readEntity(String.class));
}
}
@Override
public void sendMessageAsync(final String destAddress, final String message, final boolean command) {
getRequestBuilder().async().post(
Entity.entity(preparePayload(destAddress, message), mediaType), new InvocationCallback<String>() {
@Override
public void completed(String s) {
}
@Override
public void failed(Throwable throwable) {
LOGGER.warn("SMS send failed", throwable);
}
});
}
}
|