123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 |
- (*
- This file is part of libmicrohttpd
- Copyright (C) 2007 Christian Grothoff (and other contributing authors)
- This library is free software; you can redistribute it and/or
- modify it under the terms of the GNU Lesser General Public
- License as published by the Free Software Foundation; either
- version 2.1 of the License, or (at your option) any later version.
- This library is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- Lesser General Public License for more details.
- You should have received a copy of the GNU Lesser General Public
- License along with this library; if not, write to the Free Software
- Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- *)
- (**
- * @file fileserver_example_dirs.pp (Original: fileserver_example_dirs.c)
- * @brief example for how to use libmicrohttpd to serve files (with directory support)
- * @author Christian Grothoff / Silvio Clécio
- *)
- program fileserver_example_dirs;
- {$mode objfpc}{$H+}
- uses
- sysutils, BaseUnix, cutils, libmicrohttpd;
- const
- PAGE: Pcchar = '<html><head><title>File not found</title></head><body>File not found</body></html>';
- function file_reader(cls: Pointer; pos: cuint64; buf: Pcchar;
- max: size_t): ssize_t; cdecl;
- var
- &file: FILEptr;
- begin
- &file := cls;
- fseek(&file, pos, SEEK_SET);
- Result := fread(buf, 1, max, &file);
- end;
- procedure file_free_callback(cls: Pointer); cdecl;
- var
- &file: FILEptr;
- begin
- &file := cls;
- fclose(&file);
- end;
- procedure dir_free_callback(cls: Pointer); cdecl;
- var
- dir: pDir;
- begin
- dir := cls;
- if dir <> nil then
- FpClosedir(dir^);
- end;
- function dir_reader(cls: Pointer; pos: cuint64; buf: Pcchar;
- max: size_t): ssize_t; cdecl;
- var
- dir: pDir;
- e: pDirent;
- begin
- dir := cls;
- if max < 512 then
- Exit(0);
- repeat
- e := FpReaddir(dir^);
- if e = nil then
- Exit(MHD_CONTENT_READER_END_OF_STREAM);
- until not (e^.d_name[0] = '.');
- Result := snprintf(buf, max, '<a href="/%s">%s</a><br>', e^.d_name,
- e^.d_name);
- end;
- function ahc_echo(cls: Pointer; connection: PMHD_Connection; url: Pcchar;
- method: Pcchar; version: Pcchar; upload_data: Pcchar;
- upload_data_size: Psize_t; ptr: PPointer): cint; cdecl;
- const
- aptr: cint = 0;
- var
- response: PMHD_Response;
- ret: cint;
- &file: FILEptr;
- dir: pDir;
- buf: stat;
- emsg: array[0..1023] of AnsiChar;
- begin
- if 0 <> strcomp(method, MHD_HTTP_METHOD_GET) then
- Exit(MHD_NO); (* unexpected method *)
- if @aptr <> ptr^ then
- begin
- (* do never respond on first call *)
- ptr^ := @aptr;
- Exit(MHD_YES);
- end;
- ptr^ := nil; (* reset when done *)
- if (0 = FpStat(@url[1], buf)) and fpS_ISREG(buf.st_mode) then
- &file := fopen(@url[1], fopenread)
- else
- &file := nil;
- if &file = nil then
- begin
- dir := FpOpendir(PChar('.'));
- if dir = nil then
- begin
- (* most likely cause: more concurrent requests than
- available file descriptors / 2 *)
- snprintf(emsg, SizeOf(emsg), 'Failed to open directory `.'': %s'#10,
- strerror(errno^));
- response := MHD_create_response_from_buffer(strlen(emsg), @emsg,
- MHD_RESPMEM_MUST_COPY);
- if response = nil then
- Exit(MHD_NO);
- ret := MHD_queue_response(connection, MHD_HTTP_SERVICE_UNAVAILABLE,
- response);
- MHD_destroy_response(response);
- end
- else
- begin
- response := MHD_create_response_from_callback(cuint64(MHD_SIZE_UNKNOWN),
- 32 * 1024, @dir_reader, dir, @dir_free_callback);
- if response = nil then
- begin
- FpClosedir(dir^);
- Exit(MHD_NO);
- end;
- ret := MHD_queue_response(connection, MHD_HTTP_OK, response);
- MHD_destroy_response(response);
- end;
- end
- else
- begin
- response := MHD_create_response_from_callback(buf.st_size, 32 * 1024, (* 32k page size *)
- @file_reader, &file, @file_free_callback);
- if response = nil then
- begin
- fclose(&file);
- Exit(MHD_NO);
- end;
- ret := MHD_queue_response(connection, MHD_HTTP_OK, response);
- MHD_destroy_response(response);
- end;
- Result := ret;
- end;
- var
- d: PMHD_Daemon;
- begin
- if argc <> 2 then
- begin
- WriteLn(argv[0], ' PORT');
- Halt(1);
- end;
- d := MHD_start_daemon(MHD_USE_THREAD_PER_CONNECTION or MHD_USE_DEBUG,
- StrToInt(argv[1]), nil, nil, @ahc_echo, PAGE, MHD_OPTION_END);
- if d = nil then
- Halt(1);
- ReadLn;
- MHD_stop_daemon(d);
- end.
|