Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in / Register
Toggle navigation
M
mongoose
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Packages
Packages
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
submodule
mongoose
Commits
0456f0f5
Commit
0456f0f5
authored
Mar 16, 2016
by
Alexander Alashkin
Committed by
Marko Mikulicic
Mar 17, 2016
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix boundary problem
PUBLISHED_FROM=b78c97a337208007e1f622fde072cd59664ef76f
parent
72cf78de
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
202 additions
and
92 deletions
+202
-92
mongoose.c
mongoose.c
+202
-92
No files found.
mongoose.c
View file @
0456f0f5
...
@@ -4210,10 +4210,24 @@ struct mg_http_endpoint {
...
@@ -4210,10 +4210,24 @@ struct mg_http_endpoint {
mg_event_handler_t
handler
;
mg_event_handler_t
handler
;
};
};
enum
mg_http_multipart_stream_state
{
MPS_BEGIN
,
MPS_WAITING_FOR_BOUNDARY
,
MPS_WAITING_FOR_CHUNK
,
MPS_GOT_CHUNK
,
MPS_GOT_BOUNDARY
,
MPS_FINALIZE
,
MPS_FINISHED
};
struct
mg_http_multipart_stream
{
struct
mg_http_multipart_stream
{
const
char
*
boundary
;
const
char
*
boundary
;
int
boundary_len
;
const
char
*
var_name
;
const
char
*
var_name
;
const
char
*
file_name
;
const
char
*
file_name
;
int
prev_io_len
;
enum
mg_http_multipart_stream_state
state
;
int
processing_part
;
};
};
struct
mg_http_proto_data
{
struct
mg_http_proto_data
{
...
@@ -5027,12 +5041,10 @@ static void mg_http_call_endpoint_handler(struct mg_connection *nc, int ev,
...
@@ -5027,12 +5041,10 @@ static void mg_http_call_endpoint_handler(struct mg_connection *nc, int ev,
}
}
#ifdef MG_ENABLE_HTTP_STREAMING_MULTIPART
#ifdef MG_ENABLE_HTTP_STREAMING_MULTIPART
static
void
mg_http_multipart_continue
(
struct
mg_connection
*
nc
,
static
void
mg_http_multipart_continue
(
struct
mg_connection
*
nc
);
struct
mbuf
*
io
,
int
ev
,
void
*
ev_data
);
static
void
mg_http_multipart_begin
(
struct
mg_connection
*
nc
,
static
void
mg_http_multipart_begin
(
struct
mg_connection
*
nc
,
struct
http_message
*
hm
,
struct
mbuf
*
io
,
struct
http_message
*
hm
,
int
req_len
);
int
req_len
);
#endif
#endif
...
@@ -5106,7 +5118,7 @@ void mg_http_handler(struct mg_connection *nc, int ev, void *ev_data) {
...
@@ -5106,7 +5118,7 @@ void mg_http_handler(struct mg_connection *nc, int ev, void *ev_data) {
#ifdef MG_ENABLE_HTTP_STREAMING_MULTIPART
#ifdef MG_ENABLE_HTTP_STREAMING_MULTIPART
if
(
pd
->
mp_stream
.
boundary
!=
NULL
)
{
if
(
pd
->
mp_stream
.
boundary
!=
NULL
)
{
mg_http_multipart_continue
(
nc
,
io
,
ev
,
ev_data
);
mg_http_multipart_continue
(
nc
);
return
;
return
;
}
}
#endif
/* MG_ENABLE_HTTP_STREAMING_MULTIPART */
#endif
/* MG_ENABLE_HTTP_STREAMING_MULTIPART */
...
@@ -5210,18 +5222,25 @@ void mg_http_handler(struct mg_connection *nc, int ev, void *ev_data) {
...
@@ -5210,18 +5222,25 @@ void mg_http_handler(struct mg_connection *nc, int ev, void *ev_data) {
mbuf_remove
(
io
,
hm
->
message
.
len
);
mbuf_remove
(
io
,
hm
->
message
.
len
);
#ifdef MG_ENABLE_HTTP_STREAMING_MULTIPART
#ifdef MG_ENABLE_HTTP_STREAMING_MULTIPART
}
else
{
}
else
{
mg_http_multipart_begin
(
nc
,
hm
,
io
,
req_len
);
mg_http_multipart_begin
(
nc
,
hm
,
req_len
);
#endif
/* MG_ENABLE_HTTP_STREAMING_MULTIPART */
#endif
/* MG_ENABLE_HTTP_STREAMING_MULTIPART */
}
}
}
}
}
}
static
size_t
mg_get_line_len
(
const
char
*
buf
,
size_t
buf_len
)
{
size_t
len
=
0
;
while
(
len
<
buf_len
&&
buf
[
len
]
!=
'\n'
)
len
++
;
return
len
==
buf_len
?
0
:
len
+
1
;
}
#ifdef MG_ENABLE_HTTP_STREAMING_MULTIPART
#ifdef MG_ENABLE_HTTP_STREAMING_MULTIPART
static
void
mg_http_multipart_begin
(
struct
mg_connection
*
nc
,
static
void
mg_http_multipart_begin
(
struct
mg_connection
*
nc
,
struct
http_message
*
hm
,
struct
mbuf
*
io
,
struct
http_message
*
hm
,
int
req_len
)
{
int
req_len
)
{
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
nc
);
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
nc
);
struct
mg_str
*
ct
;
struct
mg_str
*
ct
;
struct
mbuf
*
io
=
&
nc
->
recv_mbuf
;
const
char
multipart
[]
=
"multipart"
;
const
char
multipart
[]
=
"multipart"
;
char
boundary
[
100
];
char
boundary
[
100
];
int
boundary_len
;
int
boundary_len
;
...
@@ -5266,6 +5285,7 @@ static void mg_http_multipart_begin(struct mg_connection *nc,
...
@@ -5266,6 +5285,7 @@ static void mg_http_multipart_begin(struct mg_connection *nc,
nc
->
flags
|=
MG_F_CLOSE_IMMEDIATELY
;
nc
->
flags
|=
MG_F_CLOSE_IMMEDIATELY
;
}
else
{
}
else
{
pd
->
mp_stream
.
boundary
=
strdup
(
boundary
);
pd
->
mp_stream
.
boundary
=
strdup
(
boundary
);
pd
->
mp_stream
.
boundary_len
=
strlen
(
boundary
);
pd
->
mp_stream
.
var_name
=
pd
->
mp_stream
.
file_name
=
NULL
;
pd
->
mp_stream
.
var_name
=
pd
->
mp_stream
.
file_name
=
NULL
;
pd
->
endpoint_handler
=
mg_http_get_endpoint_handler
(
nc
->
listener
,
&
hm
->
uri
);
pd
->
endpoint_handler
=
mg_http_get_endpoint_handler
(
nc
->
listener
,
&
hm
->
uri
);
...
@@ -5281,104 +5301,200 @@ exit_mp:
...
@@ -5281,104 +5301,200 @@ exit_mp:
;
;
}
}
static
void
mg_http_multipart_continue
(
struct
mg_connection
*
nc
,
#define CONTENT_DISPOSITION "Content-Disposition: "
struct
mbuf
*
io
,
int
ev
,
void
*
ev_data
)
{
/* Continue to stream multipart */
struct
mg_http_multipart_part
mp
;
const
char
*
boundary
;
int
req_len
;
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
nc
);
static
void
mg_http_multipart_call_handler
(
struct
mg_connection
*
c
,
int
ev
,
const
char
*
data
,
size_t
data_len
)
{
struct
mg_http_multipart_part
mp
;
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
c
);
memset
(
&
mp
,
0
,
sizeof
(
mp
));
memset
(
&
mp
,
0
,
sizeof
(
mp
));
mp
.
var_name
=
pd
->
mp_stream
.
var_name
;
mp
.
var_name
=
pd
->
mp_stream
.
var_name
;
mp
.
file_name
=
pd
->
mp_stream
.
file_name
;
mp
.
file_name
=
pd
->
mp_stream
.
file_name
;
boundary
=
c_strnstr
(
io
->
buf
,
pd
->
mp_stream
.
boundary
,
io
->
len
);
mp
.
data
.
p
=
data
;
if
(
boundary
==
NULL
)
{
mp
.
data
.
len
=
data_len
;
mp
.
data
.
p
=
io
->
buf
;
mg_call
(
c
,
pd
->
endpoint_handler
,
ev
,
&
mp
);
mp
.
data
.
len
=
io
->
len
;
}
mg_call
(
nc
,
pd
->
endpoint_handler
,
MG_EV_HTTP_PART_DATA
,
&
mp
);
mbuf_remove
(
io
,
io
->
len
);
}
else
{
int
has_prefix
=
0
,
has_suffix
=
0
,
boundary_len
=
strlen
(
pd
->
mp_stream
.
boundary
);
if
(
boundary
-
2
>=
io
->
buf
)
{
has_prefix
=
(
strncmp
(
boundary
-
2
,
"--"
,
2
)
==
0
);
}
if
(
boundary
+
boundary_len
<=
io
->
buf
+
io
->
len
)
{
has_suffix
=
(
strncmp
(
boundary
+
boundary_len
,
"--"
,
2
)
==
0
);
}
if
(
has_prefix
&&
!
has_suffix
)
{
/* No suffix - not last boundary */
char
varname
[
100
]
=
{
0
},
filename
[
100
]
=
{
0
};
const
char
*
data
=
NULL
;
size_t
data_len
=
0
;
int
num_left
=
(
boundary
-
io
->
buf
)
-
2
;
/* Send remainder of the previous part (if any) to callback */
if
(
num_left
>
2
)
{
/* \r\n */
mp
.
data
.
p
=
io
->
buf
;
mp
.
data
.
len
=
num_left
-
2
;
mg_call
(
nc
,
pd
->
endpoint_handler
,
MG_EV_HTTP_PART_DATA
,
&
mp
);
mp
.
data
.
len
=
0
;
mg_call
(
nc
,
pd
->
endpoint_handler
,
MG_EV_HTTP_PART_END
,
&
mp
);
mbuf_remove
(
io
,
num_left
);
}
mg_parse_multipart
(
io
->
buf
,
io
->
len
,
varname
,
sizeof
(
varname
),
filename
,
sizeof
(
filename
),
&
data
,
&
data_len
);
mp
.
var_name
=
varname
;
mp
.
file_name
=
filename
;
if
((
req_len
=
mg_http_get_request_len
(
io
->
buf
,
io
->
len
))
>
0
)
{
const
char
*
tmp
;
mg_call
(
nc
,
pd
->
endpoint_handler
,
MG_EV_HTTP_PART_BEGIN
,
&
mp
);
free
((
void
*
)
pd
->
mp_stream
.
var_name
);
pd
->
mp_stream
.
var_name
=
strdup
(
mp
.
var_name
);
free
((
void
*
)
pd
->
mp_stream
.
file_name
);
pd
->
mp_stream
.
file_name
=
strdup
(
mp
.
file_name
);
mbuf_remove
(
io
,
req_len
);
static
int
mg_http_multipart_got_chunk
(
struct
mg_connection
*
c
)
{
mp
.
data
.
p
=
io
->
buf
;
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
c
);
struct
mbuf
*
io
=
&
c
->
recv_mbuf
;
mg_http_multipart_call_handler
(
c
,
MG_EV_HTTP_PART_DATA
,
io
->
buf
,
pd
->
mp_stream
.
prev_io_len
);
mbuf_remove
(
io
,
pd
->
mp_stream
.
prev_io_len
);
pd
->
mp_stream
.
prev_io_len
=
0
;
pd
->
mp_stream
.
state
=
MPS_WAITING_FOR_CHUNK
;
return
0
;
}
tmp
=
c_strnstr
(
io
->
buf
,
pd
->
mp_stream
.
boundary
,
io
->
len
);
static
int
mg_http_multipart_finalize
(
struct
mg_connection
*
c
)
{
if
(
tmp
==
NULL
)
{
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
c
);
mp
.
data
.
len
=
io
->
len
;
mg_http_multipart_call_handler
(
c
,
MG_EV_HTTP_PART_END
,
NULL
,
0
);
mg_http_free_proto_data_mp_stream
(
&
pd
->
mp_stream
);
pd
->
mp_stream
.
state
=
MPS_FINISHED
;
return
1
;
}
static
int
mg_http_multipart_wait_for_boundary
(
struct
mg_connection
*
c
)
{
const
char
*
boundary
;
struct
mbuf
*
io
=
&
c
->
recv_mbuf
;
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
c
);
if
((
int
)
io
->
len
<
pd
->
mp_stream
.
boundary_len
+
2
)
{
return
0
;
}
boundary
=
c_strnstr
(
io
->
buf
,
pd
->
mp_stream
.
boundary
,
io
->
len
);
if
(
boundary
!=
NULL
)
{
if
(
io
->
len
-
(
boundary
-
io
->
buf
)
<
4
)
{
return
0
;
}
if
(
memcmp
(
boundary
+
pd
->
mp_stream
.
boundary_len
,
"--"
,
2
)
==
0
)
{
pd
->
mp_stream
.
state
=
MPS_FINALIZE
;
}
else
{
}
else
{
mp
.
data
.
len
=
tmp
-
io
->
buf
-
2
;
pd
->
mp_stream
.
state
=
MPS_GOT_BOUNDARY
;
}
}
else
{
return
0
;
}
}
if
(
mp
.
data
.
len
!=
0
)
{
return
1
;
size_t
data_len
=
mp
.
data
.
len
;
}
mg_call
(
nc
,
pd
->
endpoint_handler
,
MG_EV_HTTP_PART_DATA
,
&
mp
);
if
(
data_len
!=
io
->
len
)
{
static
int
mg_http_multipart_process_boundary
(
struct
mg_connection
*
c
)
{
mp
.
data
.
len
=
0
;
int
data_size
;
mg_call
(
nc
,
pd
->
endpoint_handler
,
MG_EV_HTTP_PART_END
,
&
mp
);
const
char
*
boundary
,
*
block_begin
;
struct
mbuf
*
io
=
&
c
->
recv_mbuf
;
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
c
);
char
file_name
[
100
],
var_name
[
100
];
int
line_len
;
boundary
=
c_strnstr
(
io
->
buf
,
pd
->
mp_stream
.
boundary
,
io
->
len
);
block_begin
=
boundary
+
pd
->
mp_stream
.
boundary_len
+
2
;
data_size
=
io
->
len
-
(
block_begin
-
io
->
buf
);
while
(
data_size
>
0
&&
(
line_len
=
mg_get_line_len
(
block_begin
,
data_size
))
!=
0
)
{
if
(
line_len
>
(
int
)
sizeof
(
CONTENT_DISPOSITION
)
&&
mg_ncasecmp
(
block_begin
,
CONTENT_DISPOSITION
,
sizeof
(
CONTENT_DISPOSITION
)
-
1
)
==
0
)
{
struct
mg_str
header
;
header
.
p
=
block_begin
+
sizeof
(
CONTENT_DISPOSITION
)
-
1
;
header
.
len
=
line_len
-
sizeof
(
CONTENT_DISPOSITION
)
-
1
;
mg_http_parse_header
(
&
header
,
"name"
,
var_name
,
sizeof
(
var_name
)
-
2
);
mg_http_parse_header
(
&
header
,
"filename"
,
file_name
,
sizeof
(
file_name
)
-
2
);
block_begin
+=
line_len
;
data_size
-=
line_len
;
continue
;
}
}
mbuf_remove
(
io
,
data_len
);
if
(
line_len
==
2
&&
mg_ncasecmp
(
block_begin
,
"
\r\n
"
,
2
)
==
0
)
{
mbuf_remove
(
io
,
block_begin
-
io
->
buf
+
2
);
if
(
pd
->
mp_stream
.
processing_part
!=
0
)
{
mg_http_multipart_call_handler
(
c
,
MG_EV_HTTP_PART_END
,
NULL
,
0
);
}
}
if
(
io
->
len
!=
0
)
{
free
((
void
*
)
pd
->
mp_stream
.
file_name
);
mg_http_handler
(
nc
,
ev
,
ev_data
);
pd
->
mp_stream
.
file_name
=
strdup
(
file_name
);
free
((
void
*
)
pd
->
mp_stream
.
var_name
);
pd
->
mp_stream
.
var_name
=
strdup
(
var_name
);
mg_http_multipart_call_handler
(
c
,
MG_EV_HTTP_PART_BEGIN
,
NULL
,
0
);
pd
->
mp_stream
.
state
=
MPS_WAITING_FOR_CHUNK
;
pd
->
mp_stream
.
processing_part
++
;
return
1
;
}
}
}
/* else wait for data */
}
else
if
(
has_prefix
&&
has_suffix
)
{
block_begin
+=
line_len
;
/* Last boundary */
mp
.
data
.
p
=
io
->
buf
;
mp
.
data
.
len
=
boundary
-
io
->
buf
-
4
;
if
(
mp
.
data
.
len
!=
0
)
{
mg_call
(
nc
,
pd
->
endpoint_handler
,
MG_EV_HTTP_PART_DATA
,
&
mp
);
}
}
mg_call
(
nc
,
pd
->
endpoint_handler
,
MG_EV_HTTP_PART_END
,
&
mp
)
;
pd
->
mp_stream
.
state
=
MPS_WAITING_FOR_BOUNDARY
;
/* Skip epilogue (if any) */
return
0
;
mbuf_remove
(
io
,
io
->
len
);
}
mg_http_free_proto_data_mp_stream
(
&
pd
->
mp_stream
);
static
int
mg_http_multipart_continue_wait_for_chunk
(
struct
mg_connection
*
c
)
{
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
c
);
struct
mbuf
*
io
=
&
c
->
recv_mbuf
;
const
char
*
boundary
;
if
((
int
)
io
->
len
<
pd
->
mp_stream
.
boundary_len
+
6
/* \r\n, --, -- */
)
{
return
0
;
}
boundary
=
c_strnstr
(
io
->
buf
,
pd
->
mp_stream
.
boundary
,
io
->
len
);
if
(
boundary
==
NULL
&&
pd
->
mp_stream
.
prev_io_len
==
0
)
{
pd
->
mp_stream
.
prev_io_len
=
io
->
len
;
return
0
;
}
else
if
(
boundary
==
NULL
&&
(
int
)
io
->
len
>
pd
->
mp_stream
.
prev_io_len
+
pd
->
mp_stream
.
boundary_len
+
4
)
{
pd
->
mp_stream
.
state
=
MPS_GOT_CHUNK
;
return
1
;
}
else
if
(
boundary
!=
NULL
)
{
int
data_size
=
(
boundary
-
io
->
buf
-
4
);
mg_http_multipart_call_handler
(
c
,
MG_EV_HTTP_PART_DATA
,
io
->
buf
,
data_size
);
mbuf_remove
(
io
,
(
boundary
-
io
->
buf
));
pd
->
mp_stream
.
prev_io_len
=
0
;
pd
->
mp_stream
.
state
=
MPS_WAITING_FOR_BOUNDARY
;
return
1
;
}
else
{
}
else
{
/* Malformed request */
return
0
;
nc
->
flags
|=
MG_F_CLOSE_IMMEDIATELY
;
}
DBG
((
"invalid request"
));
}
static
void
mg_http_multipart_continue
(
struct
mg_connection
*
c
)
{
struct
mg_http_proto_data
*
pd
=
mg_http_get_proto_data
(
c
);
while
(
1
)
{
switch
(
pd
->
mp_stream
.
state
)
{
case
MPS_BEGIN
:
{
pd
->
mp_stream
.
state
=
MPS_WAITING_FOR_BOUNDARY
;
break
;
}
case
MPS_WAITING_FOR_BOUNDARY
:
{
if
(
mg_http_multipart_wait_for_boundary
(
c
)
==
0
)
{
return
;
}
break
;
}
case
MPS_GOT_BOUNDARY
:
{
if
(
mg_http_multipart_process_boundary
(
c
)
==
0
)
{
return
;
}
break
;
}
case
MPS_WAITING_FOR_CHUNK
:
{
if
(
mg_http_multipart_continue_wait_for_chunk
(
c
)
==
0
)
{
return
;
}
break
;
}
case
MPS_GOT_CHUNK
:
{
if
(
mg_http_multipart_got_chunk
(
c
)
==
0
)
{
return
;
}
break
;
}
case
MPS_FINALIZE
:
{
if
(
mg_http_multipart_finalize
(
c
)
==
0
)
{
return
;
}
break
;
}
case
MPS_FINISHED
:
{
mbuf_remove
(
&
c
->
recv_mbuf
,
c
->
recv_mbuf
.
len
);
return
;
}
}
}
}
}
}
}
#endif
/* MG_ENABLE_HTTP_STREAMING_MULTIPART */
#endif
/* MG_ENABLE_HTTP_STREAMING_MULTIPART */
void
mg_set_protocol_http_websocket
(
struct
mg_connection
*
nc
)
{
void
mg_set_protocol_http_websocket
(
struct
mg_connection
*
nc
)
{
...
@@ -7526,12 +7642,6 @@ struct mg_connection *mg_connect_http(struct mg_mgr *mgr,
...
@@ -7526,12 +7642,6 @@ struct mg_connection *mg_connect_http(struct mg_mgr *mgr,
return
nc
;
return
nc
;
}
}
static
size_t
mg_get_line_len
(
const
char
*
buf
,
size_t
buf_len
)
{
size_t
len
=
0
;
while
(
len
<
buf_len
&&
buf
[
len
]
!=
'\n'
)
len
++
;
return
buf
[
len
]
==
'\n'
?
len
+
1
:
0
;
}
size_t
mg_parse_multipart
(
const
char
*
buf
,
size_t
buf_len
,
char
*
var_name
,
size_t
mg_parse_multipart
(
const
char
*
buf
,
size_t
buf_len
,
char
*
var_name
,
size_t
var_name_len
,
char
*
file_name
,
size_t
var_name_len
,
char
*
file_name
,
size_t
file_name_len
,
const
char
**
data
,
size_t
file_name_len
,
const
char
**
data
,
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment