* Multipart form content length calculation

* HTMLForm test
This commit is contained in:
Rangel Reale
2013-12-04 18:19:20 -02:00
parent 5edb90c48e
commit 0d2a839f79
12 changed files with 164 additions and 5 deletions

View File

@@ -89,6 +89,15 @@ public:
int getCurrentLineNumber() const; int getCurrentLineNumber() const;
/// Returns the current line number (same as lines()). /// Returns the current line number (same as lines()).
void addChars(int chars);
/// Add to the total number of characters.
void addLines(int lines);
/// Add to the total number of lines.
void addPos(int pos);
/// Add to the number of characters on the current line.
protected: protected:
int readFromDevice(); int readFromDevice();
int writeToDevice(char c); int writeToDevice(char c);
@@ -144,6 +153,15 @@ public:
int getCurrentLineNumber() const; int getCurrentLineNumber() const;
/// Returns the current line number (same as lines()). /// Returns the current line number (same as lines()).
void addChars(int chars);
/// Add to the total number of characters.
void addLines(int lines);
/// Add to the total number of lines.
void addPos(int pos);
/// Add to the number of characters on the current line.
CountingStreamBuf* rdbuf(); CountingStreamBuf* rdbuf();
/// Returns a pointer to the underlying streambuf. /// Returns a pointer to the underlying streambuf.

View File

@@ -116,6 +116,24 @@ void CountingStreamBuf::setCurrentLineNumber(int line)
} }
void CountingStreamBuf::addChars(int chars)
{
_chars += chars;
}
void CountingStreamBuf::addLines(int lines)
{
_lines += lines;
}
void CountingStreamBuf::addPos(int pos)
{
_pos += pos;
}
CountingIOS::CountingIOS() CountingIOS::CountingIOS()
{ {
poco_ios_init(&_buf); poco_ios_init(&_buf);
@@ -151,6 +169,24 @@ void CountingIOS::setCurrentLineNumber(int line)
} }
void CountingIOS::addChars(int chars)
{
_buf.addChars(chars);
}
void CountingIOS::addLines(int lines)
{
_buf.addLines(lines);
}
void CountingIOS::addPos(int pos)
{
_buf.addPos(pos);
}
CountingStreamBuf* CountingIOS::rdbuf() CountingStreamBuf* CountingIOS::rdbuf()
{ {
return &_buf; return &_buf;

View File

@@ -74,6 +74,16 @@ void CountingStreamTest::testInput()
assert (ci3.lines() == 2); assert (ci3.lines() == 2);
assert (ci3.chars() == 8); assert (ci3.chars() == 8);
assert (ci3.pos() == 0); assert (ci3.pos() == 0);
std::istringstream istr4("foo");
CountingInputStream ci4(istr4);
while (ci4.good()) ci4.get(c);
ci4.addChars(10);
ci4.addLines(2);
ci4.addPos(3);
assert (ci4.lines() == 1 + 2);
assert (ci4.chars() == 3 + 10);
assert (ci4.pos() == 3 + 3);
} }
@@ -100,6 +110,16 @@ void CountingStreamTest::testOutput()
assert (co3.lines() == 2); assert (co3.lines() == 2);
assert (co3.chars() == 8); assert (co3.chars() == 8);
assert (co3.pos() == 0); assert (co3.pos() == 0);
std::ostringstream ostr4;
CountingOutputStream co4(ostr4);
co4 << "foo";
co4.addChars(10);
co4.addLines(2);
co4.addPos(3);
assert (co4.lines() == 1 + 2);
assert (co4.chars() == 3 + 10);
assert (co4.pos() == 3 + 3);
} }

View File

@@ -83,7 +83,11 @@ public:
const std::string& filename() const; const std::string& filename() const;
/// Returns the filename portion of the path. /// Returns the filename portion of the path.
std::streamsize getContentLength() const;
/// Returns the file size.
private: private:
std::string _path;
std::string _filename; std::string _filename;
Poco::FileInputStream _istr; Poco::FileInputStream _istr;
}; };

View File

@@ -173,7 +173,14 @@ public:
/// Otherwise, if the request's HTTP version is HTTP/1.1: /// Otherwise, if the request's HTTP version is HTTP/1.1:
/// - the request's persistent connection state is left unchanged /// - the request's persistent connection state is left unchanged
/// - the content transfer encoding is set to chunked /// - the content transfer encoding is set to chunked
std::streamsize calculateContentLength();
/// Calculate the content length for the form.
/// May be UNKNOWN_CONTENT_LENGTH if not possible
/// to calculate
void write(std::ostream& ostr, const std::string& boundary); void write(std::ostream& ostr, const std::string& boundary);
/// Writes the form data to the given output stream, /// Writes the form data to the given output stream,
/// using the specified encoding. /// using the specified encoding.
@@ -203,6 +210,7 @@ public:
static const std::string ENCODING_URL; /// "application/x-www-form-urlencoded" static const std::string ENCODING_URL; /// "application/x-www-form-urlencoded"
static const std::string ENCODING_MULTIPART; /// "multipart/form-data" static const std::string ENCODING_MULTIPART; /// "multipart/form-data"
static const int UNKNOWN_CONTENT_LENGTH;
protected: protected:
void readUrl(std::istream& istr); void readUrl(std::istream& istr);
void readMultipart(std::istream& istr, PartHandler& handler); void readMultipart(std::istream& istr, PartHandler& handler);

View File

@@ -77,9 +77,15 @@ public:
/// Returns a MessageHeader containing additional header /// Returns a MessageHeader containing additional header
/// fields for the part. /// fields for the part.
virtual std::streamsize getContentLength() const;
/// Returns the content length for this part
/// which may be UNKNOWN_CONTENT_LENGTH if
/// not available.
virtual ~PartSource(); virtual ~PartSource();
/// Destroys the PartSource. /// Destroys the PartSource.
static const int UNKNOWN_CONTENT_LENGTH;
protected: protected:
PartSource(); PartSource();
/// Creates the PartSource, using /// Creates the PartSource, using

View File

@@ -75,6 +75,9 @@ public:
const std::string& filename() const; const std::string& filename() const;
/// Returns the filename portion of the path. /// Returns the filename portion of the path.
std::streamsize getContentLength() const;
/// Returns the string size.
private: private:
std::istringstream _istr; std::istringstream _istr;
std::string _filename; std::string _filename;

View File

@@ -36,6 +36,7 @@
#include "Poco/Net/FilePartSource.h" #include "Poco/Net/FilePartSource.h"
#include "Poco/Path.h" #include "Poco/Path.h"
#include "Poco/File.h"
#include "Poco/Exception.h" #include "Poco/Exception.h"
@@ -48,7 +49,7 @@ namespace Net {
FilePartSource::FilePartSource(const std::string& path): FilePartSource::FilePartSource(const std::string& path):
_istr(path) _path(path), _istr(path)
{ {
Path p(path); Path p(path);
_filename = p.getFileName(); _filename = p.getFileName();
@@ -59,6 +60,7 @@ FilePartSource::FilePartSource(const std::string& path):
FilePartSource::FilePartSource(const std::string& path, const std::string& mediaType): FilePartSource::FilePartSource(const std::string& path, const std::string& mediaType):
PartSource(mediaType), PartSource(mediaType),
_path(path),
_istr(path) _istr(path)
{ {
Path p(path); Path p(path);
@@ -70,6 +72,7 @@ FilePartSource::FilePartSource(const std::string& path, const std::string& media
FilePartSource::FilePartSource(const std::string& path, const std::string& filename, const std::string& mediaType): FilePartSource::FilePartSource(const std::string& path, const std::string& filename, const std::string& mediaType):
PartSource(mediaType), PartSource(mediaType),
_path(path),
_filename(filename), _filename(filename),
_istr(path) _istr(path)
{ {
@@ -96,4 +99,11 @@ const std::string& FilePartSource::filename() const
} }
std::streamsize FilePartSource::getContentLength() const
{
Poco::File p(_path);
return p.getSize();
}
} } // namespace Poco::Net } } // namespace Poco::Net

View File

@@ -47,6 +47,7 @@
#include "Poco/StreamCopier.h" #include "Poco/StreamCopier.h"
#include "Poco/URI.h" #include "Poco/URI.h"
#include "Poco/String.h" #include "Poco/String.h"
#include "Poco/CountingStream.h"
#include <sstream> #include <sstream>
@@ -61,8 +62,21 @@ namespace Poco {
namespace Net { namespace Net {
const std::string HTMLForm::ENCODING_URL = "application/x-www-form-urlencoded"; const std::string HTMLForm::ENCODING_URL = "application/x-www-form-urlencoded";
const std::string HTMLForm::ENCODING_MULTIPART = "multipart/form-data"; const std::string HTMLForm::ENCODING_MULTIPART = "multipart/form-data";
const int HTMLForm::UNKNOWN_CONTENT_LENGTH = -1;
class HTMLFormCountingOutputStream : public CountingOutputStream
{
public:
HTMLFormCountingOutputStream() : _isvalid(true) {}
bool getIsValid() const { return _isvalid; }
void setIsValid(bool v) { _isvalid = v; }
private:
bool _isvalid;
};
HTMLForm::HTMLForm(): HTMLForm::HTMLForm():
@@ -237,6 +251,20 @@ void HTMLForm::prepareSubmit(HTTPRequest& request)
} }
std::streamsize HTMLForm::calculateContentLength()
{
if (_boundary.empty())
throw HTMLFormException("Form must be prepared");
HTMLFormCountingOutputStream c;
write(c);
if (c.getIsValid())
return c.chars();
else
return UNKNOWN_CONTENT_LENGTH;
}
void HTMLForm::write(std::ostream& ostr, const std::string& boundary) void HTMLForm::write(std::ostream& ostr, const std::string& boundary)
{ {
if (_encoding == ENCODING_URL) if (_encoding == ENCODING_URL)
@@ -358,6 +386,8 @@ void HTMLForm::writeUrl(std::ostream& ostr)
void HTMLForm::writeMultipart(std::ostream& ostr) void HTMLForm::writeMultipart(std::ostream& ostr)
{ {
HTMLFormCountingOutputStream *costr(dynamic_cast<HTMLFormCountingOutputStream*>(&ostr));
MultipartWriter writer(ostr, _boundary); MultipartWriter writer(ostr, _boundary);
for (NameValueCollection::ConstIterator it = begin(); it != end(); ++it) for (NameValueCollection::ConstIterator it = begin(); it != end(); ++it)
{ {
@@ -385,7 +415,17 @@ void HTMLForm::writeMultipart(std::ostream& ostr)
header.set("Content-Disposition", disp); header.set("Content-Disposition", disp);
header.set("Content-Type", ita->pSource->mediaType()); header.set("Content-Type", ita->pSource->mediaType());
writer.nextPart(header); writer.nextPart(header);
StreamCopier::copyStream(ita->pSource->stream(), ostr); if (costr)
{
// count only, don't move stream position
std::streamsize partlen = ita->pSource->getContentLength();
if (partlen != PartSource::UNKNOWN_CONTENT_LENGTH)
costr->addChars(partlen);
else
costr->setIsValid(false);
}
else
StreamCopier::copyStream(ita->pSource->stream(), ostr);
} }
writer.close(); writer.close();
_boundary = writer.boundary(); _boundary = writer.boundary();

View File

@@ -41,6 +41,9 @@ namespace Poco {
namespace Net { namespace Net {
const int PartSource::UNKNOWN_CONTENT_LENGTH = -1;
PartSource::PartSource(): PartSource::PartSource():
_mediaType("application/octet-stream") _mediaType("application/octet-stream")
{ {
@@ -69,5 +72,9 @@ const std::string& PartSource::filename() const
return EMPTY; return EMPTY;
} }
std::streamsize PartSource::getContentLength() const
{
return UNKNOWN_CONTENT_LENGTH;
}
} } // namespace Poco::Net } } // namespace Poco::Net

View File

@@ -80,4 +80,10 @@ const std::string& StringPartSource::filename() const
} }
std::streamsize StringPartSource::getContentLength() const
{
return _istr.str().length();
}
} } // namespace Poco::Net } } // namespace Poco::Net

View File

@@ -167,6 +167,7 @@ void HTMLFormTest::testWriteMultipart()
"This is another attachment\r\n" "This is another attachment\r\n"
"--MIME_boundary_0123456789--\r\n" "--MIME_boundary_0123456789--\r\n"
); );
assert(s.length() == form.calculateContentLength());
} }