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
|
// Copyright 2015 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.
#include "components/safe_json/json_sanitizer.h"
#if defined(OS_ANDROID)
#error Build json_sanitizer_android.cc instead of this file on Android.
#endif
#include <memory>
#include "base/bind.h"
#include "base/callback.h"
#include "base/json/json_writer.h"
#include "base/macros.h"
#include "base/memory/weak_ptr.h"
#include "base/strings/string_util.h"
#include "base/values.h"
#include "build/build_config.h"
#include "components/safe_json/safe_json_parser.h"
namespace safe_json {
namespace {
class OopJsonSanitizer : public JsonSanitizer {
public:
OopJsonSanitizer(const std::string& unsafe_json,
const StringCallback& success_callback,
const StringCallback& error_callback);
private:
friend std::default_delete<OopJsonSanitizer>;
~OopJsonSanitizer() {}
void OnParseSuccess(std::unique_ptr<base::Value> value);
void OnParseError(const std::string& error);
StringCallback success_callback_;
StringCallback error_callback_;
DISALLOW_COPY_AND_ASSIGN(OopJsonSanitizer);
};
OopJsonSanitizer::OopJsonSanitizer(const std::string& unsafe_json,
const StringCallback& success_callback,
const StringCallback& error_callback)
: success_callback_(success_callback), error_callback_(error_callback) {
SafeJsonParser::Parse(unsafe_json,
base::Bind(&OopJsonSanitizer::OnParseSuccess,
base::Unretained(this)),
base::Bind(&OopJsonSanitizer::OnParseError,
base::Unretained(this)));
}
void OopJsonSanitizer::OnParseSuccess(std::unique_ptr<base::Value> value) {
// Self-destruct at the end of this method.
std::unique_ptr<OopJsonSanitizer> deleter(this);
// A valid JSON document may only have a dictionary or list as its top-level
// type, but the JSON parser also accepts other types, so we filter them out.
base::Value::Type type = value->GetType();
if (type != base::Value::Type::DICTIONARY &&
type != base::Value::Type::LIST) {
error_callback_.Run("Invalid top-level type");
return;
}
std::string json;
if (!base::JSONWriter::Write(*value, &json)) {
error_callback_.Run("Encoding error");
return;
}
success_callback_.Run(json);
}
void OopJsonSanitizer::OnParseError(const std::string& error) {
error_callback_.Run("Parse error: " + error);
delete this;
}
} // namespace
// static
void JsonSanitizer::Sanitize(const std::string& unsafe_json,
const StringCallback& success_callback,
const StringCallback& error_callback) {
// OopJsonSanitizer destroys itself when it is finished.
new OopJsonSanitizer(unsafe_json, success_callback, error_callback);
}
} // namespace safe_json
|