Old Ada http server.
parent
c019a5a884
commit
4a9d20f1ee
|
@ -0,0 +1,17 @@
|
|||
package body signals is
|
||||
protected body Sigint_Handler is
|
||||
|
||||
procedure selector(s: Selector_Access) is
|
||||
begin
|
||||
sel := s;
|
||||
end;
|
||||
|
||||
procedure Handle is
|
||||
begin
|
||||
Call_Count := Call_Count + 1;
|
||||
i Put_Line("SIGINT handled");
|
||||
Gnat.Sockets.Abort_Selector(sel.all);
|
||||
|
||||
END Handle;
|
||||
end Sigint_Handler;
|
||||
end signals;
|
|
@ -0,0 +1,13 @@
|
|||
Ala ma kota
|
||||
adasdada
|
||||
asda
|
||||
d
|
||||
|
||||
Dada
|
||||
sd
|
||||
a
|
||||
d
|
||||
a
|
||||
da
|
||||
|
||||
takaotootootootgf
|
|
@ -0,0 +1,7 @@
|
|||
package constants is
|
||||
|
||||
cr : Character := Character'Val(13);
|
||||
lf : Character := Character'Val(10);
|
||||
newline: String := (cr, lf);
|
||||
|
||||
end constants;
|
|
@ -0,0 +1,75 @@
|
|||
package body dispatchers is
|
||||
|
||||
task body Dispatcher is
|
||||
HANDLERS_COUNT : constant Integer := 2;
|
||||
MAX_SOCKETS_COUNT : constant Integer := 1000;
|
||||
UNDISPATCHED_SOCKET_RETRY_TIME : constant Duration := 0.5;
|
||||
|
||||
handlers : array (1 .. HANDLERS_COUNT) of Handler;
|
||||
|
||||
type Socket_Index is range 1..MAX_SOCKETS_COUNT;
|
||||
package Socket_Vector is new Ada.Containers.Vectors (
|
||||
Element_Type => Socket_Type,
|
||||
Index_Type => Socket_Index);
|
||||
|
||||
function Find_Free_Handler(s: Socket_Type) return Boolean is
|
||||
begin
|
||||
for id in handlers'Range loop
|
||||
select
|
||||
handlers(id).handle(s);
|
||||
return True;
|
||||
else
|
||||
null;
|
||||
end select;
|
||||
end loop;
|
||||
return False;
|
||||
end Find_Free_Handler;
|
||||
|
||||
result : Boolean;
|
||||
undispatched : Socket_Vector.Vector;
|
||||
|
||||
begin
|
||||
accept start;
|
||||
|
||||
for id in handlers'Range loop
|
||||
handlers(id).start(id);
|
||||
end loop;
|
||||
|
||||
loop
|
||||
select
|
||||
accept dispatch (s: Socket_Type) do
|
||||
Put_Line ("dispatch command");
|
||||
result := Find_Free_Handler(s);
|
||||
if not result then
|
||||
Put_Line("All handlers are busy");
|
||||
undispatched.Append(s);
|
||||
end if;
|
||||
end dispatch;
|
||||
or
|
||||
accept stop do
|
||||
Put_Line("stop command, stopping dispatcher");
|
||||
end stop;
|
||||
exit;
|
||||
or
|
||||
delay UNDISPATCHED_SOCKET_RETRY_TIME;
|
||||
if Integer(undispatched.Length) > 0 then
|
||||
declare
|
||||
s: Socket_Type := undispatched.Element(1);
|
||||
begin
|
||||
result := Find_Free_Handler(s);
|
||||
if result then
|
||||
undispatched.Delete(1);
|
||||
Put_Line("Handled undispatched client");
|
||||
end if;
|
||||
end;
|
||||
end if;
|
||||
end select;
|
||||
end loop;
|
||||
|
||||
for id in handlers'Range loop
|
||||
handlers(id).stop;
|
||||
end loop;
|
||||
|
||||
|
||||
end dispatcher;
|
||||
end dispatchers;
|
|
@ -0,0 +1,17 @@
|
|||
with Ada.Strings.Unbounded; use Ada.Strings.Unbounded;
|
||||
with Ada.Text_IO; use Ada.Text_IO;
|
||||
with Gnat.Sockets; use Gnat.Sockets;
|
||||
with Ada.Calendar; use Ada.Calendar;
|
||||
with Ada.Containers.Vectors;
|
||||
with Handlers; use Handlers;
|
||||
|
||||
package dispatchers is
|
||||
task Dispatcher is
|
||||
entry start;
|
||||
-- signal to start
|
||||
entry dispatch(s: Socket_Type);
|
||||
-- we got accepted socket from the listener
|
||||
entry stop;
|
||||
-- signal to stop
|
||||
end dispatcher;
|
||||
end dispatchers;
|
|
@ -0,0 +1,247 @@
|
|||
package body handlers is
|
||||
task body handler is
|
||||
socket : Socket_Type;
|
||||
sel : access Gnat.Sockets.Selector_Type := new
|
||||
Gnat.Sockets.Selector_Type;
|
||||
working : Boolean := True;
|
||||
subtype Socket_Index is Integer range 1 .. 1000;
|
||||
root_path : constant String := "/var/www";
|
||||
|
||||
package Socket_Vector is new Ada.Containers.Vectors (
|
||||
Element_Type => Socket_Type,
|
||||
Index_Type => Socket_Index
|
||||
);
|
||||
|
||||
v : Socket_Vector.Vector;
|
||||
|
||||
task watcher is
|
||||
entry start;
|
||||
end watcher;
|
||||
|
||||
task body watcher is
|
||||
R : Socket_Set_Type;
|
||||
W : Socket_Set_Type;
|
||||
E : Socket_Set_Type;
|
||||
status : Selector_Status;
|
||||
active : Socket_Type;
|
||||
unparsed: Unbounded_String;
|
||||
begin
|
||||
accept start;
|
||||
Put_Line("start watcher");
|
||||
Create_Selector (sel.all);
|
||||
loop
|
||||
Put_Line("Watch loop start");
|
||||
Empty(R);
|
||||
Empty(W);
|
||||
Empty(E);
|
||||
|
||||
Put("There is ");
|
||||
Put(v.Last_Index, Width => 0);
|
||||
Put_Line(" sockets to watch");
|
||||
|
||||
Put_Line("setting to read sockets");
|
||||
|
||||
for id in 1..v.Last_Index loop
|
||||
set(R, v.Element(id));
|
||||
end loop;
|
||||
|
||||
Put_Line("start of check_selector");
|
||||
|
||||
Check_Selector(
|
||||
sel.all,
|
||||
R,
|
||||
W,
|
||||
E,
|
||||
status,
|
||||
Gnat.Sockets.Forever);
|
||||
|
||||
Put_Line("end of check_selector");
|
||||
|
||||
-- wychodzimy jedynie wtedy, gdy zmienna working
|
||||
-- jest False. Inaczej to jedynie przeładowanie
|
||||
-- socketów np. dodano nam socket do obserwacji
|
||||
exit when status = Gnat.Sockets.Aborted
|
||||
and not working;
|
||||
|
||||
if status /= Gnat.Sockets.Aborted then
|
||||
Put_Line("Sending command to client handler");
|
||||
Get(R, active);
|
||||
Handler.command(active);
|
||||
end if;
|
||||
|
||||
Put_Line("Watch loop end");
|
||||
|
||||
end loop;
|
||||
Close_Selector (sel.all);
|
||||
Put_Line("exit watcher");
|
||||
end watcher;
|
||||
|
||||
procedure Handle_File(s: Socket_Type;
|
||||
path: String;
|
||||
finished: out Boolean) is
|
||||
|
||||
sz : Natural := Natural(Size(path));
|
||||
subtype Content_Type is String (1 .. sz);
|
||||
package dio is new Ada.Direct_IO(Content_Type);
|
||||
File : dio.File_Type;
|
||||
contents : Content_Type;
|
||||
channel : Stream_Access := Stream(s);
|
||||
begin
|
||||
Put_Line("Handle_File start");
|
||||
Put_Line("Path: " & path);
|
||||
Put_Line("File size: " & Natural'Image(sz));
|
||||
|
||||
dio.Open(
|
||||
File => File,
|
||||
Mode => dio.In_File,
|
||||
Name => path);
|
||||
|
||||
Put_Line("File opened");
|
||||
|
||||
String'Write(channel, "HTTP/1.0 200 OK" & newline);
|
||||
String'Write(channel, "Server: Ada Server 0.1" &
|
||||
newline);
|
||||
|
||||
Put_Line("Root path: " & root_path & path);
|
||||
Put_Line("File size: " & Natural'Image(sz));
|
||||
|
||||
dio.Read (File, Item => contents);
|
||||
Put_Line("Contents: " & contents & newline);
|
||||
String'Write(channel, newline);
|
||||
String'Write(channel, contents);
|
||||
|
||||
dio.Close(File);
|
||||
Put_Line("Closed file");
|
||||
|
||||
Close_Socket(s);
|
||||
Put_Line("Closed socket");
|
||||
finished := True;
|
||||
end Handle_File;
|
||||
|
||||
procedure Handle_Socket(s: Socket_Type; finished: out Boolean) is
|
||||
subtype Line_Type is String (1 .. 4096);
|
||||
channel : Stream_Access := Stream (s);
|
||||
line : Line_Type;
|
||||
elem_line : Stream_Element_Array (1 .. 4096);
|
||||
last : Stream_Element_Offset;
|
||||
|
||||
package af renames Ada.Strings.Fixed;
|
||||
|
||||
function Convert is new Ada.Unchecked_Conversion (
|
||||
Source => Stream_Element_Array,
|
||||
Target => Line_Type
|
||||
);
|
||||
begin
|
||||
Put_Line("Got command to execute");
|
||||
channel := Stream (s);
|
||||
Receive_Socket(s, elem_line, last);
|
||||
line := Convert(elem_line);
|
||||
Put("Odczytano: ");
|
||||
Put(Integer(last), Width => 0);
|
||||
Put_Line(" znakow");
|
||||
if last >= 4 and line (1 .. 4) = "GET " then
|
||||
Put_Line("GET Command: " & line);
|
||||
|
||||
declare
|
||||
EOL : Integer := af.Index(line, newline);
|
||||
HTTP_Version_Offset : Integer := af.Index(line, "HTTP/1.");
|
||||
path : String := line (5..HTTP_Version_Offset - 2);
|
||||
begin
|
||||
if Exists (root_path & path) then
|
||||
Put_Line("Kind: " & File_Kind'Image(Kind(root_path & path)));
|
||||
if Kind(root_path & path) = Ordinary_File then
|
||||
Put_Line("Full path: '" & root_path & path & "'");
|
||||
handle_file(s,
|
||||
root_path & path,
|
||||
finished);
|
||||
return;
|
||||
else
|
||||
-- pewnie katalog
|
||||
if Exists(root_path & path & "index.html") then
|
||||
handle_file(s,
|
||||
root_path & path &
|
||||
"index.html",
|
||||
finished);
|
||||
|
||||
return;
|
||||
end if;
|
||||
|
||||
if Exists(root_path & path & "index.htm") then
|
||||
handle_file(s,
|
||||
root_path & path &
|
||||
"index.htm",
|
||||
finished);
|
||||
|
||||
return;
|
||||
end if;
|
||||
|
||||
String'Write(channel, "HTTP/1.0 200 OK" & newline);
|
||||
String'Write(channel, "Server: Ada Server 0.1" &
|
||||
newline & newline);
|
||||
String'Write(channel, "<html><body>" &
|
||||
"directory</body></html>" & newline);
|
||||
end if;
|
||||
else
|
||||
String'Write (channel, "HTTP/1.0 404 NOT FOUND" &
|
||||
newline);
|
||||
String'Write (channel, "Server: Ada Server 0.1" &
|
||||
newline & newline);
|
||||
String'Write (channel, "<html><body>" &
|
||||
"404 Error</body></html>");
|
||||
end if;
|
||||
end;
|
||||
|
||||
else
|
||||
String'Write(channel, "Unknown" & newline);
|
||||
end if;
|
||||
Put_Line("Closing socket");
|
||||
Close_Socket(s);
|
||||
finished := True;
|
||||
end Handle_Socket;
|
||||
|
||||
idx : Integer;
|
||||
|
||||
begin
|
||||
accept start(id: Integer) do
|
||||
idx := id;
|
||||
end start;
|
||||
|
||||
Watcher.Start;
|
||||
|
||||
Put("start client handler with id: ");
|
||||
Put(idx, Width => 0);
|
||||
New_Line;
|
||||
loop
|
||||
select
|
||||
accept handle(s: Socket_Type) do
|
||||
socket := s;
|
||||
v.Append(s);
|
||||
end handle;
|
||||
Abort_Selector(sel.all);
|
||||
Put_Line("Socket consumed by client handler");
|
||||
or
|
||||
accept command(s: Socket_Type) do
|
||||
declare
|
||||
finished: Boolean;
|
||||
begin
|
||||
Handle_Socket(s, finished);
|
||||
if finished then
|
||||
v.Delete(v.Find_Index(s));
|
||||
end if;
|
||||
end;
|
||||
end command;
|
||||
or
|
||||
accept stop do
|
||||
Put_Line("Got command to stop");
|
||||
working := False;
|
||||
Abort_Selector(sel.all);
|
||||
end stop;
|
||||
exit;
|
||||
or
|
||||
terminate;
|
||||
end select;
|
||||
end loop;
|
||||
Put_Line("end client handler");
|
||||
end handler;
|
||||
|
||||
end handlers;
|
|
@ -0,0 +1,37 @@
|
|||
with Ada.Containers.Vectors;
|
||||
with Ada.Integer_Text_IO; use Ada.Integer_Text_IO;
|
||||
with Ada.Strings.Unbounded; use Ada.Strings.Unbounded;
|
||||
with Ada.Text_IO; use Ada.Text_IO;
|
||||
with GNAT.Sockets; use GNAT.Sockets;
|
||||
with Constants; use Constants;
|
||||
with Ada.Calendar; use Ada.Calendar;
|
||||
with Ada.Streams; use Ada.Streams;
|
||||
with Ada.Unchecked_Conversion;
|
||||
with Ada.Strings.Fixed;
|
||||
with Ada.Directories; use Ada.Directories;
|
||||
with Ada.Direct_IO;
|
||||
with Ada.Exceptions; use Ada.Exceptions;
|
||||
with Ada.IO_Exceptions; use Ada.IO_Exceptions;
|
||||
|
||||
package handlers is
|
||||
|
||||
type Client_Type is record
|
||||
socket : Socket_Type;
|
||||
read : Unbounded_String;
|
||||
unread : Unbounded_String;
|
||||
last_active : Time;
|
||||
end record;
|
||||
|
||||
type Client_Index_Type is range 1..100;
|
||||
package Client_Vector_Type is new Ada.Containers.Vectors (
|
||||
Element_Type => Client_Type,
|
||||
Index_Type => Client_Index_Type);
|
||||
|
||||
task type handler is
|
||||
entry start(id: Integer);
|
||||
entry handle(s: Socket_Type);
|
||||
entry command(s: Socket_Type);
|
||||
entry stop;
|
||||
end handler;
|
||||
|
||||
end handlers;
|
Binary file not shown.
|
@ -0,0 +1,67 @@
|
|||
with Ada.Text_IO; use Ada.Text_IO;
|
||||
with Ada.Integer_Text_IO; use Ada.Integer_Text_IO;
|
||||
with GNAT.Sockets; use GNAT.Sockets;
|
||||
with signals; use signals;
|
||||
with Ada.Containers.Vectors;
|
||||
with Handlers; use Handlers;
|
||||
with Constants; use Constants;
|
||||
with Dispatchers; use Dispatchers;
|
||||
|
||||
procedure server is
|
||||
|
||||
HOST : constant String := "localhost";
|
||||
PORT : constant Port_Type := 6666;
|
||||
work : Boolean := True;
|
||||
address : Sock_Addr_Type;
|
||||
server : Socket_Type;
|
||||
socket : Socket_Type;
|
||||
h2 : Sigint_Handler;
|
||||
status : Gnat.Sockets.Selector_Status;
|
||||
selector: signals.selector_Access;
|
||||
req : Request_Type(Non_Blocking_IO);
|
||||
begin
|
||||
Put_Line("start of server");
|
||||
Dispatcher.Start;
|
||||
selector := new Gnat.Sockets.Selector_Type;
|
||||
h2.selector(selector);
|
||||
Gnat.Sockets.Create_Selector(selector.all);
|
||||
address.Addr := Addresses (Get_Host_By_Name(host), 1);
|
||||
address.Port := PORT;
|
||||
|
||||
Create_Socket(server);
|
||||
Set_Socket_Option (
|
||||
server,
|
||||
Socket_Level,
|
||||
(Reuse_Address, True));
|
||||
|
||||
Bind_Socket (server, address);
|
||||
Put_Line("Binded");
|
||||
Listen_Socket(server);
|
||||
Put_Line("Listening");
|
||||
|
||||
loop
|
||||
Put_Line("Waiting to accept");
|
||||
Accept_Socket (
|
||||
Server => server,
|
||||
Socket => socket,
|
||||
Address => address,
|
||||
Timeout => GNAT.Sockets.Forever,
|
||||
Selector => selector,
|
||||
Status => status);
|
||||
exit when status = Gnat.Sockets.Aborted;
|
||||
Put_Line("Accepted");
|
||||
|
||||
Control_Socket(socket, req);
|
||||
|
||||
Dispatcher.dispatch(socket);
|
||||
Put_Line("Handled by listener");
|
||||
|
||||
end loop;
|
||||
|
||||
Gnat.Sockets.Close_Selector(selector.all);
|
||||
Close_Socket (server);
|
||||
|
||||
Dispatcher.Stop;
|
||||
|
||||
Put_Line("end of server");
|
||||
end server;
|
|
@ -0,0 +1,16 @@
|
|||
package body signals is
|
||||
protected body Sigint_Handler is
|
||||
|
||||
procedure selector(s: Selector_Access) is
|
||||
begin
|
||||
sel := s;
|
||||
end;
|
||||
|
||||
procedure Handle is
|
||||
begin
|
||||
Call_Count := Call_Count + 1;
|
||||
Put_Line("SIGINT handled");
|
||||
Gnat.Sockets.Abort_Selector(sel.all);
|
||||
end Handle;
|
||||
end Sigint_Handler;
|
||||
end signals;
|
|
@ -0,0 +1,23 @@
|
|||
with Ada.Interrupts; use Ada.Interrupts;
|
||||
with Ada.Interrupts.Names; use Ada.Interrupts.Names;
|
||||
with Ada.Text_IO; use Ada.Text_IO;
|
||||
with Gnat.Sockets; use Gnat.Sockets;
|
||||
|
||||
package signals is
|
||||
|
||||
pragma Unreserve_All_Interrupts;
|
||||
|
||||
type Selector_Access is access all Gnat.Sockets.Selector_Type;
|
||||
|
||||
protected type Sigint_Handler is
|
||||
procedure selector(s: Selector_Access);
|
||||
procedure Handle;
|
||||
|
||||
pragma Interrupt_Handler(Handle);
|
||||
pragma Attach_Handler(Handle, Sigint);
|
||||
private
|
||||
Call_Count : Natural := 0;
|
||||
sel : Selector_Access;
|
||||
end Sigint_Handler;
|
||||
|
||||
end signals;
|
|
@ -0,0 +1,13 @@
|
|||
Ala ma kota
|
||||
adasdada
|
||||
asda
|
||||
d
|
||||
|
||||
Dada
|
||||
sd
|
||||
a
|
||||
d
|
||||
a
|
||||
da
|
||||
|
||||
taka
|
Loading…
Reference in New Issue