View Javadoc
1   /**
2    * Copyright 2011-2014 The Kuali Foundation Licensed under the
3    * Educational Community License, Version 2.0 (the "License"); you may
4    * not use this file except in compliance with the License. You may
5    * obtain a copy of the License at
6    *
7    * http://www.osedu.org/licenses/ECL-2.0
8    *
9    * Unless required by applicable law or agreed to in writing,
10   * software distributed under the License is distributed on an "AS IS"
11   * BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
12   * or implied. See the License for the specific language governing
13   * permissions and limitations under the License.
14   */
15  
16  package org.kuali.mobility.writer.service;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.apache.log4j.Logger;
20  import org.kuali.mobility.email.service.EmailService;
21  import org.kuali.mobility.push.entity.Device;
22  import org.kuali.mobility.push.entity.Preference;
23  import org.kuali.mobility.push.entity.Push;
24  import org.kuali.mobility.push.service.DeviceService;
25  import org.kuali.mobility.push.service.PreferenceService;
26  import org.kuali.mobility.push.service.PushService;
27  import org.kuali.mobility.security.user.api.User;
28  import org.kuali.mobility.writer.entity.Article;
29  import org.springframework.beans.factory.annotation.Autowired;
30  import org.springframework.context.MessageSource;
31  
32  import javax.annotation.Resource;
33  import java.sql.Timestamp;
34  import java.util.ArrayList;
35  import java.util.List;
36  import java.util.Locale;
37  import java.util.Properties;
38  import java.util.concurrent.ExecutorService;
39  import java.util.concurrent.Executors;
40  
41  /**
42   * This class is responsible for publishing writer articles.
43   * This implementation will create notification for users that has to be
44   * notified of a published article, and will also send an email to the
45   * editor of the article.
46   *
47   * @author Kuali Mobility Team (mobility.collab@kuali.org)
48   * @since 3.1
49   */
50  public class PublishServiceImpl implements PublishService {
51  
52  	/**
53  	 * A reference to a logger
54  	 */
55  	private Logger LOG = Logger.getLogger(PublishServiceImpl.class);
56  
57  	/**
58  	 * A cached thread pool which allows us to queue notification tasks
59  	 */
60  	private final ExecutorService exeService = Executors.newCachedThreadPool();
61  
62  	/**
63  	 * A reference to the push service
64  	 */
65  	@Autowired
66  	private PushService pushService;
67  
68  	/**
69  	 * A reference to the <code>EmailService</code>.
70  	 */
71  	@Autowired
72  	private EmailService emailService;
73  	
74  	/**
75  	 * A reference to the <code>MessageSource</code>
76  	 */
77  	@Resource(name="messageSource")
78  	private MessageSource messageSource;
79  
80  	/**
81  	 * A reference to the writer tool's properties
82  	 */
83  	@Resource(name="writerProperties")
84  	private Properties writerProperties;
85  
86  	/**
87  	 * A reference to the <code>PreferenceService</code>
88  	 */
89  	@Autowired
90  	private PreferenceService preferenceService;
91  
92  	/**
93  	 * A reference to the device service.
94  	 */
95  	@Autowired
96  	private DeviceService deviceService;
97  
98  	@Override
99  	public void publishArticle(Article article, User user) {
100 		this.exeService.execute(new PublishRunner(article, user));
101 	}
102 
103 	/**
104 	 * Find the users that should be notified about a published article.
105 	 * This implementation will find all users that did not opt out for push notifications for this tool's instance.
106 	 *
107 	 * You could override this method for your institution if you have other ways to find students to sent notifications
108 	 * for.
109 	 */
110 	protected List<String> findUsersToNotify(Article article){
111 		List<String> users = new ArrayList<String>();
112 
113 		// Get the sender id for the instance of the tool
114 		String senderId = getInstanceSenderId(article.getToolInstance());
115 
116 		// If there is no sender ID this tool instance may not send push notifications
117 		if(StringUtils.isEmpty(senderId)){
118 			LOG.info("This tool instance does not have a sender ID, therefore it may not send push notifications");
119 			return users; // List is this empty
120 		}
121 
122 		// Get all the registered devices
123 		List<Device> devices = deviceService.findAllDevices();
124 
125 		// Now find out who did not opt out for push notifications
126 		for(Device device : devices){
127 			String username = device.getUsername();
128 			Preference preference = preferenceService.findPreference(username, senderId);
129 			if(!users.contains(username) && (preference == null || !preference.isSenderBlocked())){
130 				users.add(username);
131 			}
132 		}
133 
134 		return users;
135 	}
136 
137 	private String getInstanceSenderId(String instance){
138 	   String property = "writer.senderId." + instance.toLowerCase();
139 	   return writerProperties.getProperty(property);
140 	}
141 
142 	/**
143 	 * A runnable that will be used to send the notification using a device
144 	 * specific implementation of the send service.
145 	 */
146 	private final class PublishRunner implements Runnable{
147 
148 		/** Article being published */
149 		private final Article article;
150 
151 		/** User that is publishing the article */
152 		private final User user;
153 
154 		/**
155 		 * 
156 		 * @param article
157 		 * @param user
158 		 */
159 		PublishRunner(Article article, User user){
160 			this.article = article;
161 			this.user = user;
162 		}
163 
164 		/*
165 		 * (non-Javadoc)
166 		 * @see java.lang.Runnable#run()
167 		 */
168 		@Override
169 		public void run() {
170 			String instance = this.article.getToolInstance();
171 			String senderKey = getInstanceSenderId(instance);
172 			if (senderKey == null){
173 				LOG.warn("Not sending push notifications. Expected to have property \"writer.senderId." + instance.toLowerCase() + "\" set in writer.config.properties");
174 				return;
175 			}
176 			List<String> usernames = findUsersToNotify(article);
177 			Push push = sendPush(senderKey, this.article, usernames);
178 
179             // Send email to the user if the user has an email
180             if(!StringUtils.isEmpty(user.getEmail())){
181                 sendMail(push);
182             }
183 
184 		}
185 		
186 		/**
187 		 * Creates and send the Push message
188 		 * @param article Article being published
189 		 * @param usernames Usernames to send notification to.
190 		 * @return
191 		 */
192 		private Push sendPush(String senderKey, Article article, List<String> usernames){
193 			String toolInstance = this.article.getToolInstance();
194 			Push push = new Push();
195 			push.setEmergency(article.getCategory() == Article.CATEGORY_IMPORTANT);
196 			push.setMessage(article.getSynopsis());
197 			push.setPostedTimestamp(new Timestamp(System.currentTimeMillis()));
198 			push.setSender(senderKey);
199 			push.setTitle(article.getHeading());
200 			push.setUrl("/writer/"+toolInstance+"/viewArticle?articleId=" + article.getId());
201 			push.setRecipients(usernames.size());
202 			pushService.savePush(push);
203 			pushService.sendPushToUsers(push, usernames);
204 			return push;
205 		}
206 
207 		/**
208 		 * Sends email to the editor of the article
209 		 * @param push
210 		 */
211 		private void sendMail(Push push){
212 			Locale locale =  new Locale(user.getPreferredLanguage());
213 			String emailTo = user.getEmail();
214 			String displayName = user.getLoginName();
215 			Object[] values = new Object[] { 
216 					displayName, // user display name
217 					article.getHeading(), // Article heading
218 					messageSource.getMessage(article.getTopic().getLabel(), null, locale), // Topic
219 					messageSource.getMessage(getCategoryLabel(), null,locale), // Category
220 					article.getSynopsis(), // Synopsis
221 					article.getText(), // Article text
222 					push.getTitle(), 	// Push title
223 					push.getRecipients(), // Push recipients
224 					push.getMessage()};
225 			
226 			String emailSubject = messageSource.getMessage("writer.publish.email.subject", null, locale);
227 			String emailBody = messageSource.getMessage("writer.publish.email.body", values, locale);
228 			emailService.sendEmail(emailBody, emailSubject, emailTo, null);
229 		}
230 		
231 		private String getCategoryLabel(){
232 			long category = article.getCategory();
233 			if (category == Article.CATEGORY_GENERAL){
234 				return "writer.common";
235 			}
236 			else {
237 				return "writer.important";
238 			}
239 		}
240 
241 	}
242 }