summaryrefslogtreecommitdiffstats
path: root/chrome/browser/extensions/api/web_request/upload_data_presenter.h
blob: 3c1f6ff38ecaf53f6cc3d92798d198e278e1b20d (plain)
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
// Copyright (c) 2012 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 CHROME_BROWSER_EXTENSIONS_API_WEB_REQUEST_UPLOAD_DATA_PRESENTER_H_
#define CHROME_BROWSER_EXTENSIONS_API_WEB_REQUEST_UPLOAD_DATA_PRESENTER_H_

#include <string>
#include <vector>

#include "base/gtest_prod_util.h"
#include "base/memory/scoped_ptr.h"

namespace base {
class DictionaryValue;
class ListValue;
class Value;
}

namespace extensions {
class FormDataParser;
}

namespace net {
class URLRequest;
class UploadElement;
}

namespace extensions {

FORWARD_DECLARE_TEST(WebRequestUploadDataPresenterTest, RawData);

// UploadDataPresenter is an interface for objects capable to consume a series
// of UploadElement and represent this data as a base:Value.
//
// Workflow for objects implementing this interface:
// 1. Call object->FeedNext(element) for each element from the request's body.
// 2. Check if object->Succeeded().
// 3. If that check passed then retrieve object->Result().
class UploadDataPresenter {
 public:
  virtual ~UploadDataPresenter();
  virtual void FeedNext(const net::UploadElement& element) = 0;
  virtual bool Succeeded() = 0;
  virtual scoped_ptr<base::Value> Result() = 0;

 protected:
  UploadDataPresenter() {}

 private:
  DISALLOW_COPY_AND_ASSIGN(UploadDataPresenter);
};

// This class passes all the bytes from elements of TYPE_BYTES as a BinaryValue
// for each such element. Elements of TYPE_FILE are presented as StringValue
// containing the path for that file.
class RawDataPresenter : public UploadDataPresenter {
 public:
  RawDataPresenter();
  virtual ~RawDataPresenter();

  // Implementation of UploadDataPresenter.
  virtual void FeedNext(const net::UploadElement& element) OVERRIDE;
  virtual bool Succeeded() OVERRIDE;
  virtual scoped_ptr<base::Value> Result() OVERRIDE;

  // Appends a dictionary {'key': 'value'} to |list|.
  // This is a helper function and has nothing to do with RawDataPresenter
  // directly. However, it is used by FeedNext* methods and also unit-tests
  // depending on RawDataPresenter. It is here to avoid code duplication.
  static void AppendResultWithKey(
      base::ListValue* list, const char* key, base::Value* value);

 private:
  // Clears resources and the success flag.
  void Abort();

  void FeedNextBytes(const char* bytes, size_t size);
  void FeedNextFile(const std::string& filename);
  FRIEND_TEST_ALL_PREFIXES(WebRequestUploadDataPresenterTest, RawData);

  bool success_;
  scoped_ptr<base::ListValue> list_;
};

// This class inspects the contents of elements of TYPE_BYTES. It uses the
// parser classes inheriting from FormDataParser to parse the concatenated
// content of such elements. If the parsing is successfull, the parsed form is
// returned as a DictionaryValue. For example, a form consisting of
// <input name="check" type="checkbox" value="A" checked />
// <input name="check" type="checkbox" value="B" checked />
// <input name="text" type="text" value="abc" />
// would be represented as {"check": ["A", "B"], "text": ["abc"]} (although as a
// DictionaryValue, not as a JSON string).
class ParsedDataPresenter : public UploadDataPresenter {
 public:
  explicit ParsedDataPresenter(const net::URLRequest* request);
  virtual ~ParsedDataPresenter();

  // Implementation of UploadDataPresenter.
  virtual void FeedNext(const net::UploadElement& element) OVERRIDE;
  virtual bool Succeeded() OVERRIDE;
  virtual scoped_ptr<base::Value> Result() OVERRIDE;

 private:
  // Clears resources and the success flag.
  void Abort();
  scoped_ptr<FormDataParser> parser_;
  bool success_;
  scoped_ptr<base::DictionaryValue> dictionary_;
};

}  // namespace extensions

#endif  // CHROME_BROWSER_EXTENSIONS_API_WEB_REQUEST_UPLOAD_DATA_PRESENTER_H_