/src/main/java/org/lwes/emitter/DatagramSocketEventEmitter.java

http://github.com/lwes/lwes-java · Java · 283 lines · 126 code · 35 blank · 122 comment · 21 complexity · f4ad7e778d1402b92da782784d8034b2 MD5 · raw file

  1. /*======================================================================*
  2. * Copyright (c) 2015, OpenX. All rights reserved. *
  3. * *
  4. * Licensed under the New BSD License (the "License"); you may not use *
  5. * this file except in compliance with the License. Unless required *
  6. * by applicable law or agreed to in writing, software distributed *
  7. * under the License is distributed on an "AS IS" BASIS, WITHOUT *
  8. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. *
  9. * See the License for the specific language governing permissions and *
  10. * limitations under the License. See accompanying LICENSE file. *
  11. *======================================================================*/
  12. package org.lwes.emitter;
  13. import org.apache.commons.logging.Log;
  14. import org.apache.commons.logging.LogFactory;
  15. import org.lwes.Event;
  16. import org.lwes.EventFactory;
  17. import org.lwes.EventSystemException;
  18. import org.lwes.util.NumberCodec;
  19. import java.io.IOException;
  20. import java.io.InputStream;
  21. import java.net.DatagramPacket;
  22. import java.net.DatagramSocket;
  23. import java.net.InetAddress;
  24. /**
  25. * Abstract class for emitting events on a datagram socket.
  26. *
  27. * @author Kenneth Kharma
  28. * @since 2.1.0
  29. */
  30. public abstract class DatagramSocketEventEmitter<T extends DatagramSocket>
  31. extends AbstractEventEmitter {
  32. private static transient Log log =
  33. LogFactory.getLog(DatagramSocketEventEmitter.class);
  34. /* the socket being used */
  35. protected T socket = null;
  36. /* the address */
  37. protected InetAddress address = null;
  38. /* the port */
  39. protected int port = 9191;
  40. /* the socket interface */
  41. protected InetAddress iface = null;
  42. /* a lock variable to synchronize events */
  43. protected Object lock = new Object();
  44. /**
  45. * Default constructor.
  46. */
  47. public DatagramSocketEventEmitter() {
  48. super();
  49. }
  50. public DatagramSocketEventEmitter(EventFactory factory) {
  51. super(factory);
  52. }
  53. /**
  54. * Sets the destination address for this emitter.
  55. *
  56. * @param address the address
  57. *
  58. */
  59. public void setAddress(InetAddress address) {
  60. this.address = address;
  61. }
  62. /**
  63. * Gets the address for this emitter.
  64. *
  65. * @return the address
  66. */
  67. public InetAddress getAddress() {
  68. return this.address;
  69. }
  70. /**
  71. * Sets the destination port for this emitter.
  72. *
  73. * @param port the port
  74. *
  75. */
  76. public void setPort(int port) {
  77. this.port = port;
  78. }
  79. /**
  80. * Gets the destination port for this emitter.
  81. *
  82. * @return the port
  83. */
  84. public int getPort() {
  85. return this.port;
  86. }
  87. /**
  88. * Sets the network interface for this emitter.
  89. *
  90. * @param iface the network interface
  91. */
  92. public void setInterface(InetAddress iface) {
  93. this.iface = iface;
  94. }
  95. /**
  96. * Gets the network interface for this emitter.
  97. *
  98. * @return the interface address
  99. */
  100. public InetAddress getInterface() {
  101. return this.iface;
  102. }
  103. /**
  104. * Sets the ESF file used for event validation.
  105. * @param esfFilePath the path of the ESF file
  106. */
  107. public void setESFFilePath(String esfFilePath) {
  108. if (getFactory() != null) {
  109. getFactory().setESFFilePath(esfFilePath);
  110. }
  111. }
  112. /**
  113. * Gets the ESF file used for event validation
  114. * @return the ESF file path
  115. */
  116. public String getESFFilePath() {
  117. if (getFactory() != null) {
  118. return getFactory().getESFFilePath();
  119. } else {
  120. return null;
  121. }
  122. }
  123. /**
  124. * Sets an InputStream to be used for event validation.
  125. * @param esfInputStream an InputStream used for event validation
  126. */
  127. public void setESFInputStream(InputStream esfInputStream) {
  128. if (getFactory() != null) {
  129. getFactory().setESFInputStream(esfInputStream);
  130. }
  131. }
  132. /**
  133. * Gets the InputStream being used for event validation.
  134. * @return the InputStream of the ESF validator
  135. */
  136. public InputStream getESFInputStream() {
  137. if (getFactory() != null) {
  138. return getFactory().getESFInputStream();
  139. } else {
  140. return null;
  141. }
  142. }
  143. /**
  144. * Creates the underlying datagram socket.
  145. * @throws IOException if an I/O error occurs during socket creation.
  146. */
  147. protected abstract void createSocket() throws IOException;
  148. /**
  149. * Initializes the emitter.
  150. * @throws IOException if an I/O error occurs during initialization.
  151. */
  152. @Override
  153. public void initialize() throws IOException {
  154. createSocket();
  155. super.initialize();
  156. }
  157. /**
  158. * @throws IOException if an I/O error occurs on shutdown.
  159. */
  160. public void shutdown() throws IOException {
  161. // close the socket AFTER calling super shutdown since that is trying to
  162. // send a shutdown message.
  163. super.shutdown();
  164. if (socket != null) {
  165. socket.close();
  166. }
  167. }
  168. /**
  169. * Creates a new event named <tt>eventName</tt>.
  170. * @param eventName the name of the event to be created
  171. * @return a new Event
  172. * @exception EventSystemException if there is a problem creating the event
  173. */
  174. public Event createEvent(String eventName) throws EventSystemException {
  175. return createEvent(eventName, true);
  176. }
  177. /**
  178. * Creates a new event named <tt>eventName</tt>.
  179. * @param eventName the name of the event to be created
  180. * @param validate whether or not to validate the event against the EventTemplateDB
  181. * @return a new Event
  182. * @exception EventSystemException if there is a problem creating the event
  183. */
  184. public Event createEvent(String eventName, boolean validate) throws EventSystemException {
  185. if (getFactory() != null) {
  186. return getFactory().createEvent(eventName, validate);
  187. } else {
  188. throw new EventSystemException("EventFactory not initialized");
  189. }
  190. }
  191. /**
  192. * Emits the event to the network.
  193. *
  194. * @param event the event to emit
  195. * @exception IOException throws an IOException is there is a network error.
  196. * @throws EventSystemException if unable to serialize the event
  197. * @return number of bytes emitted
  198. */
  199. public int emit(Event event) throws IOException, EventSystemException {
  200. byte[] msg = event.serialize();
  201. int bytesEmitted = 0;
  202. synchronized (lock) {
  203. bytesEmitted = emit(msg);
  204. try {
  205. collectStatistics();
  206. }
  207. catch (EventSystemException e) {
  208. log.error(e.getMessage(), e);
  209. }
  210. }
  211. return bytesEmitted;
  212. }
  213. /**
  214. * Emits a byte array to the network.
  215. *
  216. * @param bytes the byte array to emit
  217. * @exception IOException throws an IOException if there is a network error.
  218. * @return number of bytes emitted
  219. */
  220. protected int emit(byte[] bytes) throws IOException {
  221. return emit(bytes, this.address, this.port);
  222. }
  223. /**
  224. * @param bytes the byte array to emit
  225. * @param address the address to use
  226. * @param port the port to use
  227. * @throws IOException throws an IOException if there is a network error
  228. * @return number of bytes emitted
  229. */
  230. protected int emit(byte[] bytes, InetAddress address, int port) throws IOException {
  231. /* don't send null bytes */
  232. if (bytes == null) return 0;
  233. if (socket == null || socket.isClosed()) {
  234. throw new IOException("Socket wasn't initialized or was closed.");
  235. }
  236. DatagramPacket dp = new DatagramPacket(bytes, bytes.length, address, port);
  237. socket.send(dp);
  238. if (log.isTraceEnabled()) {
  239. log.trace("Sent to network '" +
  240. NumberCodec.byteArrayToHexString(dp.getData(), 0, dp.getLength()));
  241. }
  242. return bytes.length;
  243. }
  244. @Override
  245. public String toString() {
  246. return getClass().getSimpleName() + " [" + getAddress() + ":" + getPort() + "]";
  247. }
  248. }