// Copyright (c) 2013 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 "tools/gn/value.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_util.h" #include "tools/gn/scope.h" Value::Value() : type_(NONE), boolean_value_(false), int_value_(0), origin_(nullptr) { } Value::Value(const ParseNode* origin, Type t) : type_(t), boolean_value_(false), int_value_(0), origin_(origin) { } Value::Value(const ParseNode* origin, bool bool_val) : type_(BOOLEAN), boolean_value_(bool_val), int_value_(0), origin_(origin) { } Value::Value(const ParseNode* origin, int64 int_val) : type_(INTEGER), boolean_value_(false), int_value_(int_val), origin_(origin) { } Value::Value(const ParseNode* origin, std::string str_val) : type_(STRING), string_value_(), boolean_value_(false), int_value_(0), origin_(origin) { string_value_.swap(str_val); } Value::Value(const ParseNode* origin, const char* str_val) : type_(STRING), string_value_(str_val), boolean_value_(false), int_value_(0), origin_(origin) { } Value::Value(const ParseNode* origin, scoped_ptr scope) : type_(SCOPE), string_value_(), boolean_value_(false), int_value_(0), scope_value_(scope.Pass()), origin_(origin) { } Value::Value(const Value& other) : type_(other.type_), string_value_(other.string_value_), boolean_value_(other.boolean_value_), int_value_(other.int_value_), list_value_(other.list_value_), origin_(other.origin_) { if (type() == SCOPE && other.scope_value_.get()) scope_value_ = other.scope_value_->MakeClosure(); } Value::~Value() { } Value& Value::operator=(const Value& other) { type_ = other.type_; string_value_ = other.string_value_; boolean_value_ = other.boolean_value_; int_value_ = other.int_value_; list_value_ = other.list_value_; if (type() == SCOPE && other.scope_value_.get()) scope_value_ = other.scope_value_->MakeClosure(); origin_ = other.origin_; return *this; } // static const char* Value::DescribeType(Type t) { switch (t) { case NONE: return "none"; case BOOLEAN: return "boolean"; case INTEGER: return "integer"; case STRING: return "string"; case LIST: return "list"; case SCOPE: return "scope"; default: NOTREACHED(); return "UNKNOWN"; } } void Value::SetScopeValue(scoped_ptr scope) { DCHECK(type_ == SCOPE); scope_value_ = scope.Pass(); } std::string Value::ToString(bool quote_string) const { switch (type_) { case NONE: return ""; case BOOLEAN: return boolean_value_ ? "true" : "false"; case INTEGER: return base::Int64ToString(int_value_); case STRING: if (quote_string) { std::string escaped = string_value_; // First escape all special uses of a backslash. ReplaceSubstringsAfterOffset(&escaped, 0, "\\$", "\\\\$"); ReplaceSubstringsAfterOffset(&escaped, 0, "\\\"", "\\\\\""); // Now escape special chars. ReplaceSubstringsAfterOffset(&escaped, 0, "$", "\\$"); ReplaceSubstringsAfterOffset(&escaped, 0, "\"", "\\\""); return "\"" + escaped + "\""; } return string_value_; case LIST: { std::string result = "["; for (size_t i = 0; i < list_value_.size(); i++) { if (i > 0) result += ", "; result += list_value_[i].ToString(true); } result.push_back(']'); return result; } case SCOPE: { Scope::KeyValueMap scope_values; scope_value_->GetCurrentScopeValues(&scope_values); if (scope_values.empty()) return std::string("{ }"); std::string result = "{\n"; for (const auto& pair : scope_values) { result += " " + pair.first.as_string() + " = " + pair.second.ToString(true) + "\n"; } result += "}"; return result; } } return std::string(); } bool Value::VerifyTypeIs(Type t, Err* err) const { if (type_ == t) return true; *err = Err(origin(), std::string("This is not a ") + DescribeType(t) + ".", std::string("Instead I see a ") + DescribeType(type_) + " = " + ToString(true)); return false; } bool Value::operator==(const Value& other) const { if (type_ != other.type_) return false; switch (type_) { case Value::BOOLEAN: return boolean_value() == other.boolean_value(); case Value::INTEGER: return int_value() == other.int_value(); case Value::STRING: return string_value() == other.string_value(); case Value::LIST: if (list_value().size() != other.list_value().size()) return false; for (size_t i = 0; i < list_value().size(); i++) { if (list_value()[i] != other.list_value()[i]) return false; } return true; case Value::SCOPE: // Scopes are always considered not equal because there's currently // no use case for comparing them, and it requires a bunch of complex // iteration code. return false; default: return false; } } bool Value::operator!=(const Value& other) const { return !operator==(other); }