|  | // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #ifndef NET_FTP_FTP_CTRL_RESPONSE_BUFFER_H_ | 
|  | #define NET_FTP_FTP_CTRL_RESPONSE_BUFFER_H_ | 
|  |  | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/containers/queue.h" | 
|  | #include "base/macros.h" | 
|  | #include "net/base/net_export.h" | 
|  | #include "net/log/net_log_with_source.h" | 
|  |  | 
|  | namespace net { | 
|  |  | 
|  | struct NET_EXPORT_PRIVATE FtpCtrlResponse { | 
|  | static const int kInvalidStatusCode; | 
|  |  | 
|  | FtpCtrlResponse(); | 
|  | FtpCtrlResponse(const FtpCtrlResponse& other); | 
|  | ~FtpCtrlResponse(); | 
|  |  | 
|  | int status_code;                 // Three-digit status code. | 
|  | std::vector<std::string> lines;  // Response lines, without CRLFs. | 
|  | }; | 
|  |  | 
|  | class NET_EXPORT_PRIVATE FtpCtrlResponseBuffer { | 
|  | public: | 
|  | FtpCtrlResponseBuffer(const NetLogWithSource& net_log); | 
|  | ~FtpCtrlResponseBuffer(); | 
|  |  | 
|  | // Called when data is received from the control socket. Returns error code. | 
|  | int ConsumeData(const char* data, int data_length); | 
|  |  | 
|  | bool ResponseAvailable() const { | 
|  | return !responses_.empty(); | 
|  | } | 
|  |  | 
|  | // Returns the next response. It is an error to call this function | 
|  | // unless ResponseAvailable returns true. | 
|  | FtpCtrlResponse PopResponse(); | 
|  |  | 
|  | private: | 
|  | struct ParsedLine { | 
|  | ParsedLine(); | 
|  | ParsedLine(const ParsedLine& other); | 
|  |  | 
|  | // Indicates that this line begins with a valid 3-digit status code. | 
|  | bool has_status_code; | 
|  |  | 
|  | // Indicates that this line has the dash (-) after the code, which | 
|  | // means a multiline response. | 
|  | bool is_multiline; | 
|  |  | 
|  | // Indicates that this line could be parsed as a complete and valid | 
|  | // response line, without taking into account preceding lines (which | 
|  | // may change its meaning into a continuation of the previous line). | 
|  | bool is_complete; | 
|  |  | 
|  | // Part of response parsed as status code. | 
|  | int status_code; | 
|  |  | 
|  | // Part of response parsed as status text. | 
|  | std::string status_text; | 
|  |  | 
|  | // Text before parsing, without terminating CRLF. | 
|  | std::string raw_text; | 
|  | }; | 
|  |  | 
|  | static ParsedLine ParseLine(const std::string& line); | 
|  |  | 
|  | void ExtractFullLinesFromBuffer(); | 
|  |  | 
|  | // We keep not-yet-parsed data in a string buffer. | 
|  | std::string buffer_; | 
|  |  | 
|  | base::queue<ParsedLine> lines_; | 
|  |  | 
|  | // True if we are in the middle of parsing a multi-line response. | 
|  | bool multiline_; | 
|  |  | 
|  | // When parsing a multiline response, we don't know beforehand if a line | 
|  | // will have a continuation. So always store last line of multiline response | 
|  | // so we can append the continuation to it. | 
|  | std::string line_buf_; | 
|  |  | 
|  | // Keep the response data while we add all lines to it. | 
|  | FtpCtrlResponse response_buf_; | 
|  |  | 
|  | // As we read full responses (possibly multiline), we add them to the queue. | 
|  | base::queue<FtpCtrlResponse> responses_; | 
|  |  | 
|  | NetLogWithSource net_log_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(FtpCtrlResponseBuffer); | 
|  | }; | 
|  |  | 
|  | }  // namespace net | 
|  |  | 
|  | #endif  // NET_FTP_FTP_CTRL_RESPONSE_BUFFER_H_ |