8db9a9114503bd397c1b13ef25207fcaabadd50c
[iot2.git] / iotjava / iotrmi / Java / IoTSocket.java
1 package iotrmi.Java;
2
3 // Java libraries
4 import java.io.*;
5 import java.net.*;
6 import java.awt.*;
7 import java.util.*;
8 import java.nio.ByteBuffer;
9
10
11 /** Class IoTSocket is the basic class for IoT RMI
12  *  socket communication. This class will be extended
13  *  by both IoTSocketServer and IoTSocketClient
14  *  <p>
15  *  Adapted from Java/C++ socket implementation
16  *  by Keith Vertanen
17  *  @see        <a href="https://www.keithv.com/software/socket/</a>
18  *
19  * @author      Rahmadi Trimananda <rtrimana @ uci.edu>
20  * @version     1.0
21  * @since       2016-08-17
22  */
23 public abstract class IoTSocket {
24
25         /**
26          * Class Properties
27          */
28         protected byte data[];
29         protected int port;
30         protected Socket sock;
31         protected BufferedInputStream input;
32         protected BufferedOutputStream output;
33
34         /**
35          * Class Constant
36          */
37         //protected static int BUFFSIZE = 128000;       // how many bytes our incoming buffer can hold (original)
38         protected static int BUFFSIZE = 8388608;        // 8388608 = 2^23 bytes of memory (8MB) - this is required by our IHome speaker driver
39         protected static int MSG_LEN_SIZE = 4;  // send length in the size of integer (4 bytes)
40
41         /**
42          * Default constructor
43          */
44         protected IoTSocket(int _port) throws IOException
45         {
46                 port = _port;
47                 data = new byte[BUFFSIZE];
48         }
49
50
51         /**
52          * sendBytes() sends an array of bytes
53          */
54         public synchronized void sendBytes(byte vals[]) throws IOException
55         {
56                 int len = vals.length;
57                 // Write the length first - convert to array of 4 bytes
58                 ByteBuffer bb = ByteBuffer.allocate(MSG_LEN_SIZE);
59                 bb.putInt(len);
60                 output.write(bb.array(), 0, MSG_LEN_SIZE);
61                 output.flush();
62                 // Write the byte array
63                 output.write(vals, 0, len);
64                 output.flush();
65                 receiveAck();
66                 sendAck();
67         }
68
69
70         /**
71          * receiveBytes() receives an array of bytes
72          */
73         public synchronized byte[] receiveBytes(byte val[]) throws IOException
74         {
75                 int i;
76                 int totalbytes = 0;
77                 int numbytes;
78
79                 // Wait until input is available
80                 while(input.available() == 0);
81                 // Read the maxlen first - read 4 bytes here
82                 byte[] lenBytes = new byte[MSG_LEN_SIZE];
83                 input.read(lenBytes, 0, MSG_LEN_SIZE);
84                 int maxlen = ByteBuffer.wrap(lenBytes).getInt();
85                 // Receive until maxlen
86                 if (maxlen>BUFFSIZE) {
87                         System.out.println("IoTSocketClient/Server: Sending more bytes then will fit in buffer! Number of bytes: " + maxlen);
88                 }
89                 val = new byte[maxlen];
90                 while (totalbytes < maxlen)
91                 {
92                         numbytes = input.read(data);
93                         // copy the bytes into the result buffer
94                         for (i=totalbytes; i<totalbytes+numbytes; i++)
95                                 val[i] = data[i-totalbytes];
96                         totalbytes += numbytes;
97                 }
98                 // we now send an acknowledgement to the server to let them
99                 // know we've got it
100                 sendAck();
101                 receiveAck();
102
103                 return val;
104         }
105
106
107         /**
108          * Close socket connection
109          */
110         public synchronized void close() throws IOException
111         {
112                 sock.close();
113         }
114
115
116         /**
117          * Send ACK
118          */
119         public synchronized void sendAck() throws IOException
120         {
121                 int ack;
122                 ack = 0;
123                 output.write(ack);
124                 output.flush();
125         }
126
127
128         /**
129          * Receive ACK
130          */
131         public synchronized void receiveAck() throws IOException
132         {
133                 int ack;
134                 ack = (int) input.read();
135         }
136 }