-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathecho.ha
More file actions
114 lines (104 loc) · 3.37 KB
/
echo.ha
File metadata and controls
114 lines (104 loc) · 3.37 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
use fmt;
use net;
use net::tcp;
use net::ip;
use io;
use os;
use bufio;
use unix::poll;
use strings;
use encoding::utf8;
// My very first Hare project: A simple TCP echo server.
// The websocket stuff will follow later (if it ever does).
def MAX_CLIENTS: size = 31;
export fn main() void = {
fmt::println("Hello, World!")!;
const listener = match (tcp::listen(ip::LOCAL_V4, 8000)) {
case let socket: net::socket => yield socket;
case let err: net::error =>
fmt::fatalf("Cannot listen on 8000: {}", net::strerror(err));
};
let pollfds: [MAX_CLIENTS + 1]poll::pollfd = [
// pollfds[0] is the listener socket.
poll::pollfd {
fd = listener,
events = (poll::event::POLLIN | poll::event::POLLPRI),
revents = 0
},
// All other FDs are for client connections.
poll::pollfd { fd = 0, events = 0, revents = 0 }
...
];
let free_slots: [MAX_CLIENTS]size = [0...];
let free_slots: []size = free_slots[0..0];
defer free(free_slots);
for (let i: size = 1; i < MAX_CLIENTS + 1; i += 1) {
append(free_slots, i);
};
for (true) {
poll::poll(pollfds, poll::INDEF)!;
if (pollfds[0].revents == poll::event::POLLIN) {
// Handle a new connection.
let conn = net::accept(listener)!;
if (len(free_slots) == 0) {
fmt::fatal("Max. clients reached!");
};
// Search a new slot/free fd.
let slot = free_slots[len(free_slots) - 1];
delete(free_slots[len(free_slots) - 1]);
assert(1 <= slot && slot <= MAX_CLIENTS);
fmt::printfln("New connection: slot={}!", slot)!;
pollfds[slot] = poll::pollfd {
fd = conn,
events = (poll::event::POLLIN | poll::event::POLLPRI |
poll::event::POLLOUT | poll::event::POLLERR |
poll::event::POLLHUP),
revents = 0
};
};
for (let i: size = 1; i < MAX_CLIENTS + 1; i += 1) {
// Skip over slots without connections.
if (pollfds[i].events == 0) continue;
// Skip if nothing happend for this FD.
if (pollfds[i].revents == 0) continue;
let peer: (ip::addr, u16) = tcp::peeraddr(pollfds[i].fd) as (ip::addr, u16);
let buf: [os::BUFSZ]u8 = [0...];
let response = match (io::read(pollfds[i].fd, buf)) {
case io::EOF =>
fmt::printfln("Disconnect: {}:{} (slot={})",
ip::string(peer.0), peer.1, i)!;
io::close(pollfds[i].fd)!;
pollfds[i] = poll::pollfd { fd = 0, events = 0, revents = 0 };
append(free_slots, i);
continue;
case let err: io::error =>
fmt::printfln("Disconnect because of error: {}:{} (slot={}): {}",
ip::string(peer.0), peer.1, i, io::strerror(err))!;
io::close(pollfds[i].fd)!;
pollfds[i] = poll::pollfd { fd = 0, events = 0, revents = 0 };
append(free_slots, i);
continue;
case let n: size =>
let msg = match (strings::fromutf8(buf[0..n])) {
case let msg: str => yield msg;
case let err: encoding::utf8::invalid => yield "<invalid UTF8>";
};
fmt::printfln("Message by: {}:{} (slot={}): {}",
ip::string(peer.0), peer.1, i, msg)!;
yield msg;
};
// TODO: This could also be done using I/O multiplexing.
match (io::writeall(pollfds[i].fd, strings::toutf8(response))) {
case let err: io::error => {
fmt::printfln("Disconnect because of error: {}:{} (slot={}): {}",
ip::string(peer.0), peer.1, i, io::strerror(err))!;
io::close(pollfds[i].fd)!;
pollfds[i] = poll::pollfd { fd = 0, events = 0, revents = 0 };
append(free_slots, i);
yield;
};
case let n: size => yield;
};
};
};
};