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
115
116
117
118
119
120
121
122
123
124
125
126
|
%%
%% %CopyrightBegin%
%%
%% Copyright Ericsson AB 1997-2010. All Rights Reserved.
%%
%% The contents of this file are subject to the Erlang Public License,
%% Version 1.1, (the "License"); you may not use this file except in
%% compliance with the License. You should have received a copy of the
%% Erlang Public License along with this software. If not, it can be
%% retrieved online at http://www.erlang.org/.
%%
%% Software distributed under the License is distributed on an "AS IS"
%% basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
%% the License for the specific language governing rights and limitations
%% under the License.
%%
%% %CopyrightEnd%
%%
%%
-module(mod_get).
-export([do/1]).
-include("httpd.hrl").
-include("httpd_internal.hrl").
%% do
do(Info) ->
?DEBUG("do -> entry",[]),
case Info#mod.method of
"GET" ->
case proplists:get_value(status, Info#mod.data) of
%% A status code has been generated!
{_StatusCode, _PhraseArgs, _Reason} ->
{proceed,Info#mod.data};
%% No status code has been generated!
undefined ->
case proplists:get_value(response, Info#mod.data) of
%% No response has been generated!
undefined ->
do_get(Info);
%% A response has been generated or sent!
_Response ->
{proceed,Info#mod.data}
end
end;
%% Not a GET method!
_ ->
{proceed,Info#mod.data}
end.
do_get(Info) ->
?DEBUG("do_get -> Request URI: ~p",[Info#mod.request_uri]),
Path = mod_alias:path(Info#mod.data, Info#mod.config_db,
Info#mod.request_uri),
send_response(Info#mod.socket,Info#mod.socket_type, Path, Info).
%% The common case when no range is specified
send_response(_Socket, _SocketType, Path, Info)->
%% Send the file!
%% Find the modification date of the file
case file:open(Path,[raw,binary]) of
{ok, FileDescriptor} ->
{FileInfo, LastModified} = get_modification_date(Path),
?DEBUG("do_get -> FileDescriptor: ~p",[FileDescriptor]),
Suffix = httpd_util:suffix(Path),
MimeType = httpd_util:lookup_mime_default(Info#mod.config_db,
Suffix,"text/plain"),
%% FileInfo = file:read_file_info(Path),
Size = integer_to_list(FileInfo#file_info.size),
Headers = case Info#mod.http_version of
"HTTP/1.1" ->
[{content_type, MimeType},
{etag, httpd_util:create_etag(FileInfo)},
{content_length, Size}|LastModified];
%% OTP-4935
_ ->
%% i.e http/1.0 and http/0.9
[{content_type, MimeType},
{content_length, Size}|LastModified]
end,
send(Info, 200, Headers, FileDescriptor),
file:close(FileDescriptor),
{proceed,[{response,{already_sent,200,
FileInfo#file_info.size}},
{mime_type,MimeType}|Info#mod.data]};
{error, Reason} ->
Status = httpd_file:handle_error(Reason, "open", Info, Path),
{proceed,
[{status, Status}| Info#mod.data]}
end.
%% send
send(#mod{socket = Socket, socket_type = SocketType} = Info,
StatusCode, Headers, FileDescriptor) ->
?DEBUG("send -> send header",[]),
httpd_response:send_header(Info, StatusCode, Headers),
send_body(SocketType,Socket,FileDescriptor).
send_body(SocketType,Socket,FileDescriptor) ->
case file:read(FileDescriptor,?FILE_CHUNK_SIZE) of
{ok,Binary} ->
?DEBUG("send_body -> send another chunk: ~p",[size(Binary)]),
case httpd_socket:deliver(SocketType,Socket,Binary) of
socket_closed ->
?LOG("send_body -> socket closed while sending",[]),
socket_close;
_ ->
send_body(SocketType,Socket,FileDescriptor)
end;
eof ->
?DEBUG("send_body -> done with this file",[]),
eof
end.
get_modification_date(Path)->
{ok, FileInfo0} = file:read_file_info(Path),
LastModified =
case catch httpd_util:rfc1123_date(FileInfo0#file_info.mtime) of
Date when is_list(Date) -> [{last_modified, Date}];
_ -> []
end,
{FileInfo0, LastModified}.
|