1 #include <ccan/oserver/oserver.h>
2 #include <ccan/read_write_all/read_write_all.h>
3 #include <ccan/opt/opt.h>
4 #include <ccan/tevent/tevent.h>
5 #include <ccan/array_size/array_size.h>
7 #include <sys/socket.h>
8 #include <netinet/in.h>
9 #include <netinet/tcp.h>
22 RECEIVING_USER_QUESTION,
24 SENDING_OTHER_QUESTION_PREFIX,
25 SENDING_OTHER_QUESTION,
26 RECEIVING_OTHER_ANSWER,
28 SENDING_ANSWER_PREFIX,
33 static uint16_t state_flag_map[] = {
34 [SENDING_GREETING] = TEVENT_FD_WRITE,
35 [RECEIVING_USER_QUESTION] = TEVENT_FD_READ,
36 [AWAITING_A_SUBCLIENT] = 0,
37 [SENDING_OTHER_QUESTION_PREFIX] = TEVENT_FD_WRITE,
38 [SENDING_OTHER_QUESTION] = TEVENT_FD_WRITE,
39 [RECEIVING_OTHER_ANSWER] = TEVENT_FD_READ,
40 [AWAITING_OUR_ORACLE] = 0,
41 [SENDING_ANSWER_PREFIX] = TEVENT_FD_WRITE,
42 [SENDING_ANSWER] = TEVENT_FD_WRITE,
47 /* What are we doing today, brain? */
49 /* Our event info, and the file descriptor. */
50 struct tevent_fd *fde;
52 /* The question we read from client. */
54 /* The answer to the client. */
56 /* How many bytes of the reply we sent so far. */
59 struct oserver *oserver;
60 /* Whose question this client is answering. */
61 struct client *subclient;
62 /* Who is answering our question. */
63 struct client *oracle;
67 /* 5 clients should be enough for anybody! */
68 struct client *clients[5];
70 struct tevent_fd *fde;
73 static ssize_t write_string(int fd, const char *str)
75 return write(fd, str, strlen(str));
78 static ssize_t read_string(int fd, char **buf)
80 ssize_t ret, len, maxlen;
83 maxlen = talloc_array_length(*buf);
85 if (maxlen < len + 100) {
87 *buf = talloc_realloc(NULL, *buf, char, maxlen);
90 ret = read(fd, *buf + len, maxlen - len - 1);
92 (*buf)[len + ret] = '\0';
96 static bool input_finished(const char *str)
98 return strchr(str, '\n');
101 /* Update state, and set our READ/WRITE flags appropriately. */
102 static void set_state(struct client *c, enum state state)
105 tevent_fd_set_flags(c->fde, state_flag_map[state]);
108 /* Returns false on error, increments state on finishing string. */
109 static bool send_string(struct client *c, const char *str)
111 ssize_t len = write_string(c->fd, str + c->bytes_sent);
114 c->bytes_sent += len;
115 if (c->bytes_sent == strlen(str)) {
117 set_state(c, c->state+1);
122 static bool get_subclient(struct client *me)
126 for (i = 0; i < ARRAY_SIZE(me->oserver->clients); i++) {
127 struct client *c = me->oserver->clients[i];
130 if (c->oracle == NULL && input_finished(c->question)) {
139 static bool get_oracle(struct client *me)
143 for (i = 0; i < ARRAY_SIZE(me->oserver->clients); i++) {
144 struct client *c = me->oserver->clients[i];
147 if (c->subclient == NULL && input_finished(c->question)) {
156 static void service_client(struct tevent_context *ev,
157 struct tevent_fd *fde, uint16_t flags, void *_c)
159 struct client *c = _c;
163 case SENDING_GREETING:
164 if (!send_string(c, "Welcome. Please ask your question.\n"))
167 case RECEIVING_USER_QUESTION:
168 len = read_string(c->fd, &c->question);
171 if (input_finished(c->question)) {
172 /* Look for someone to be oracle to. */
173 if (get_subclient(c)) {
174 set_state(c, SENDING_OTHER_QUESTION_PREFIX);
176 /* We sit here until someone find_oracles us */
177 set_state(c, AWAITING_A_SUBCLIENT);
180 /* Look for an oracle for ourselves. */
182 assert(c->oracle->state
183 == AWAITING_A_SUBCLIENT);
185 SENDING_OTHER_QUESTION_PREFIX);
189 case SENDING_OTHER_QUESTION_PREFIX:
190 if (!send_string(c, "While the Oracle ponders,"
191 " please answer the following question:\n"))
194 case SENDING_OTHER_QUESTION:
195 if (!send_string(c, c->subclient->question))
198 case RECEIVING_OTHER_ANSWER:
199 len = read_string(c->fd, &c->subclient->answer);
202 if (input_finished(c->subclient->answer)) {
203 /* Did our oracle answer for us already? */
204 if (input_finished(c->answer))
205 set_state(c, SENDING_ANSWER_PREFIX);
207 set_state(c, AWAITING_OUR_ORACLE);
209 /* If they were waiting for an answer, move them. */
210 if (c->subclient->state == AWAITING_OUR_ORACLE)
211 set_state(c->subclient, SENDING_ANSWER_PREFIX);
214 case SENDING_ANSWER_PREFIX:
215 if (!send_string(c, "The Oracle spake thus:\n"))
219 if (!send_string(c, c->answer))
226 if (c->state != FINISHED)
233 static int cleanup_client(struct client *client)
237 for (i = 0; i < ARRAY_SIZE(client->oserver->clients); i++) {
238 if (client->oserver->clients[i] == client) {
239 client->oserver->clients[i] = NULL;
240 tevent_fd_set_flags(client->oserver->fde,
248 static void add_client(struct tevent_context *ev,
249 struct tevent_fd *fde, uint16_t flags, void *_oserver)
251 struct oserver *oserver = _oserver;
252 struct client *client;
255 client = talloc(oserver, struct client);
256 client->fd = accept(oserver->fd, NULL, 0);
258 err(1, "Accepting client connection");
260 client->state = SENDING_GREETING;
261 client->bytes_sent = 0;
262 client->question = talloc_strdup(client, "");
263 client->oserver = oserver;
264 client->oracle = NULL;
265 client->subclient = NULL;
266 client->answer = talloc_strdup(client, "");
267 client->fde = tevent_add_fd(ev, client, client->fd,
268 state_flag_map[client->state],
269 service_client, client);
270 tevent_fd_set_auto_close(client->fde);
272 /* Find empty slot in array for this client. */
273 for (i = 0; oserver->clients[i]; i++);
274 oserver->clients[i] = client;
275 talloc_set_destructor(client, cleanup_client);
277 /* Full? Stop listening... */
278 if (i == ARRAY_SIZE(oserver->clients)-1)
279 tevent_fd_set_flags(oserver->fde, 0);
282 static void clear_clients(struct oserver *oserver)
284 memset(oserver->clients, 0,
285 ARRAY_SIZE(oserver->clients) * sizeof(oserver->clients[0]));
288 static int destroy_oserver(struct oserver *oserver)
294 struct oserver *oserver_setup(struct tevent_context *ev, unsigned short port)
296 struct oserver *oserver;
299 struct sockaddr addr;
300 struct sockaddr_in in;
303 oserver = talloc(ev, struct oserver);
304 clear_clients(oserver);
305 oserver->fd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
306 if (oserver->fd < 0) {
307 talloc_free(oserver);
311 talloc_set_destructor(oserver, destroy_oserver);
313 if (setsockopt(oserver->fd, SOL_SOCKET, SO_REUSEADDR,
315 warn("Setting socket reuse");
317 u.in.sin_family = AF_INET;
318 u.in.sin_port = htons(port);
319 u.in.sin_addr.s_addr = INADDR_ANY;
320 if (bind(oserver->fd, &u.addr, sizeof(u.in)) == -1) {
321 talloc_free(oserver);
325 if (listen(oserver->fd, 0) != 0) {
326 talloc_free(oserver);
330 oserver->fde = tevent_add_fd(ev, oserver, oserver->fd,
331 TEVENT_FD_READ, add_client, oserver);
333 talloc_free(oserver);
337 /* Don't kill us if client dies. */
338 signal(SIGPIPE, SIG_IGN);