View Javadoc

1   /**
2    * Copyright 2011-2013 The Kuali Foundation Licensed under the Educational
3    * Community License, Version 2.0 (the "License"); you may not use this file
4    * except in compliance with the License. You may obtain a copy of the License
5    * at
6    *
7    * http://www.osedu.org/licenses/ECL-2.0
8    *
9    * Unless required by applicable law or agreed to in writing, software
10   * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11   * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12   * License for the specific language governing permissions and limitations under
13   * the License.
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   * An implementation of the SendService for iOS.
39   * 
40   * This implementation makes use of a connection pool that needs to be configured with spring.
41   * The connection pool ensures that a maximum number of open connections can be controlled, as well as closing
42   * unsed connections when not used for a period of time.
43   * <br>
44   * This implementation will attempt to send a message with three retries before giving up.
45   *
46  
47   * @author Kuali Mobility Team (mobility.dev@kuali.org)
48   * @since 2.2.0
49   */
50  public class iOSSendService implements SendService {
51  	
52  	/**
53  	 *  Template for an emergency message
54  	 */
55  	private static final String TEMPLATE_EMERGENCY = "\'{\'\"aps\":\'{\'\"alert\":\"{0}\",\"badge\":{1}\'}\',\"i\":{2},\"e\":\"{3}\",\"url\":\"{4}\"\'}\'";
56  	
57  	/**
58  	 * Template for a normal push message
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  	/** Connection pool */
71  	@Autowired
72  	private GenericObjectPool<SSLSocket> iOSConnectionPool;
73  
74  	/** Reference to a logger */
75  	private static final Logger LOG = Logger.getLogger(iOSSendService.class);
76  
77  	/**
78  	 * Sends the specified <code>Push</code> message to the specified <code>Device</code>.
79  	 * This implementation makes use of a connection pool. If there is currently no connection 
80  	 * available the current thread will block until a connection becomes available (unless 
81  	 * otherwise configured)
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); 					// Command Byte: New format = 1
92  			baos.write(deviceToken[FIRST_BYTE]); 	// Identifier Byte 1
93  			baos.write(deviceToken[SECOND_BYTE]); 	// Identifier Byte 2
94  			baos.write(deviceToken[THIRD_BYTE]); 	// Identifier Byte 3
95  			baos.write(deviceToken[FORTH_BYTE]); 	// Identifier Byte 4
96  			baos.write(0); 					// Expiry Byte 1
97  			baos.write(0); 					// Expiry Byte 2
98  			baos.write(0); 					// Expiry Byte 3
99  			baos.write(1); 					// Expiry Byte 4
100 			baos.write(0); 					// Device ID Length
101 			baos.write(DEVICE_ID_LENGTH);
102 			baos.write(deviceToken); 		// Device ID
103 			baos.write(0); 					// Payload Length
104 			baos.write(payload.length);
105 			baos.write(payload); 			// Payload
106 		} catch ( IOException e ) {
107 			LOG.error("Failed Creating Payload", e);
108 			return;
109 		} 
110 
111 		int retryAttempt = 1; // Number of tries to send the notification, quits when zero or lower
112 		boolean success = false;
113 		OutputStream out = null;//CodeReview could use chained streams here
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(); // We do not close the output stream as it is reused
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); // Close potentially broken stream
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 	 * Sends the specified push message to the list of devices.
142 	 * This implementation does not currently support sending messages to batches, 
143 	 * therefor calling this method will call <code>sendPush(Push , Device)</code>
144 	 * once for every device.
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 	 * Creates a device token from the device object
157 	 * @param device The device to create a token of
158 	 * @return The byte array of the token
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 	 * This method returns a String in the format required by APNS.
173 	 * @param p The push message to create payload of.
174 	 * @return A string formatted ready to be sent to APNS
175 	 * @throws UnsupportedEncodingException 
176 	 */
177 	private static byte[] preparePayload(Push p){
178 		String message;
179 		if (p.getEmergency()){
180 			Object[] arguments = {
181 					p.getTitle(),							// Message
182 					"1",									// Badge
183 					String.valueOf(p.getPushId()),			// Message id
184 					(p.getEmergency()?"YES":"NO"),			// Emergency
185 					(p.getUrl().length() > 0 ?"YES":"NO")	// Has url, CodeReview Might be a good idea to explain (using comments) why you do this.
186 			};
187 			message = MessageFormat.format(TEMPLATE_EMERGENCY, arguments);
188 		}
189 		else {
190 			Object[] arguments = {
191 					"1",									// Badge
192 					String.valueOf(p.getPushId()),			// Message id
193 					(p.getEmergency()?"YES":"NO"),			// Emergency
194 					(p.getUrl().length() > 0 ?"YES":"NO")	// Has url, CodeReview Might be a good idea to explain (using comments) why you do this.
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 }