1
2
3
4
5
6
7
8
9
10
11
12
13
14
15 package org.kuali.mobility.push.service.send;
16
17 import java.io.ByteArrayOutputStream;
18 import java.io.IOException;
19 import java.io.OutputStream;
20 import java.io.UnsupportedEncodingException;
21 import java.text.MessageFormat;
22 import java.util.List;
23
24 import javax.net.ssl.SSLSocket;
25
26 import org.apache.commons.codec.DecoderException;
27 import org.apache.commons.codec.binary.Hex;
28 import org.apache.commons.io.IOUtils;
29 import org.apache.commons.pool.impl.GenericObjectPool;
30 import org.apache.log4j.Level;
31 import org.apache.log4j.Logger;
32 import org.kuali.mobility.push.entity.Device;
33 import org.kuali.mobility.push.entity.Push;
34 import org.kuali.mobility.push.service.SendService;
35 import org.springframework.beans.factory.annotation.Autowired;
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50 public class iOSSendService implements SendService {
51
52
53
54
55 private static final String TEMPLATE_EMERGENCY = "\'{\'\"aps\":\'{\'\"alert\":\"{0}\",\"badge\":{1}\'}\',\"i\":{2},\"e\":\"{3}\",\"url\":\"{4}\"\'}\'";
56
57
58
59
60 private static final String TEMPLATE = "\'{\'\"aps\":\'{\'\"badge\":{0}\'}\',\"i\":{1},\"e\":\"{2}\",\"url\":\"{3}\"\'}\'";
61
62 private static final int FIRST_BYTE = 0;
63 private static final int SECOND_BYTE = 1;
64 private static final int THIRD_BYTE = 2;
65 private static final int FORTH_BYTE = 3;
66 private static final int DEVICE_ID_LENGTH = 32;
67 private static final int MAX_RETRY_ATTEMPTS = 3;
68
69
70
71 @Autowired
72 private GenericObjectPool<SSLSocket> iOSConnectionPool;
73
74
75 private static final Logger LOG = Logger.getLogger(iOSSendService.class);
76
77
78
79
80
81
82
83 @Override
84 public void sendPush( Push push, Device device ) {
85
86 byte[] payload = preparePayload(push);
87 byte[] deviceToken = createDeviceToken(device);
88
89 ByteArrayOutputStream baos = new ByteArrayOutputStream();
90 try {
91 baos.write(1);
92 baos.write(deviceToken[FIRST_BYTE]);
93 baos.write(deviceToken[SECOND_BYTE]);
94 baos.write(deviceToken[THIRD_BYTE]);
95 baos.write(deviceToken[FORTH_BYTE]);
96 baos.write(0);
97 baos.write(0);
98 baos.write(0);
99 baos.write(1);
100 baos.write(0);
101 baos.write(DEVICE_ID_LENGTH);
102 baos.write(deviceToken);
103 baos.write(0);
104 baos.write(payload.length);
105 baos.write(payload);
106 } catch ( IOException e ) {
107 LOG.error("Failed Creating Payload", e);
108 return;
109 }
110
111 int retryAttempt = 1;
112 boolean success = false;
113 OutputStream out = null;
114 while (!success && retryAttempt<=MAX_RETRY_ATTEMPTS) {
115 SSLSocket socket = null;
116 try {
117 socket= iOSConnectionPool.borrowObject();
118 out = socket.getOutputStream();
119 out.write(baos.toByteArray());
120 if (LOG.getLevel() == Level.DEBUG){
121 LOG.debug(baos.toString());
122 }
123 out.flush();
124 success = true;
125 } catch (Exception e) {
126 LOG.error("Exception while trying to write message over socket (Retry attempt : " + retryAttempt + ")", e);
127 IOUtils.closeQuietly(out);
128 retryAttempt++;
129 }
130 finally{
131 try {
132 iOSConnectionPool.returnObject(socket);
133 } catch (Exception e) {
134 LOG.warn("Exception while trying to put Socket back into pool",e);
135 }
136 }
137 }
138 }
139
140
141
142
143
144
145
146 @Override
147 public void sendPush(Push push, List<Device> devices) {
148 if (devices != null){
149 for (Device device : devices){
150 this.sendPush(push, device);
151 }
152 }
153 }
154
155
156
157
158
159
160 private static byte[] createDeviceToken(Device device){
161 char[] t = device.getRegId().toCharArray();
162 byte[] tokenBytes = null;
163 try {
164 tokenBytes = Hex.decodeHex(t);
165 } catch (DecoderException e) {
166 LOG.error("Failed decoding Token", e);
167 }
168 return tokenBytes;
169 }
170
171
172
173
174
175
176
177 private static byte[] preparePayload(Push p){
178 String message;
179 if (p.getEmergency()){
180 Object[] arguments = {
181 p.getTitle(),
182 "1",
183 String.valueOf(p.getPushId()),
184 (p.getEmergency()?"YES":"NO"),
185 (p.getUrl().length() > 0 ?"YES":"NO")
186 };
187 message = MessageFormat.format(TEMPLATE_EMERGENCY, arguments);
188 }
189 else {
190 Object[] arguments = {
191 "1",
192 String.valueOf(p.getPushId()),
193 (p.getEmergency()?"YES":"NO"),
194 (p.getUrl().length() > 0 ?"YES":"NO")
195 };
196 message = MessageFormat.format(TEMPLATE, arguments);
197 }
198 try {
199 return message.getBytes("UTF-8");
200 } catch (UnsupportedEncodingException e) {
201 LOG.warn("Exception while converting device token from string to bytes", e);
202 return null;
203 }
204 }
205
206 }