blob: 8dd4c7388b404fffdc3ca08c91dd416dab83e381 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
#ifndef T_STRUCT_H
#define T_STRUCT_H
#include <algorithm>
#include <vector>
#include <utility>
#include <string>
#include "t_type.h"
#include "t_field.h"
// Forward declare that puppy
class t_program;
/**
* A struct is a container for a set of member fields that has a name. Structs
* are also used to implement exception types.
*
*/
class t_struct : public t_type {
public:
typedef std::vector<t_field*> members_type;
t_struct(t_program* program) :
t_type(program),
is_xception_(false),
is_union_(false),
members_validated(false),
members_with_value(0),
xsd_all_(false) {}
t_struct(t_program* program, const std::string& name) :
t_type(program, name),
is_xception_(false),
is_union_(false),
members_validated(false),
members_with_value(0),
xsd_all_(false) {}
void set_name(const std::string& name) {
name_ = name;
validate_union_members();
}
void set_xception(bool is_xception) {
is_xception_ = is_xception;
}
void validate_union_member( t_field * field) {
if( is_union_ && (! name_.empty())) {
// unions can't have required fields
if( field->get_req() == t_field::T_REQUIRED) {
pwarning( 1, "Required field %s of union %s set to optional.\n", field->get_name().c_str(), name_.c_str());
field->set_req( t_field::T_OPTIONAL);
}
// unions may have up to one member defaulted, but not more
if( field->get_value() != NULL) {
if( 1 < ++members_with_value) {
throw "Error: Field "+field->get_name()+" provides another default value for union "+name_;
}
}
}
}
void validate_union_members() {
if( is_union_ && (! name_.empty()) && (!members_validated)) {
members_type::const_iterator m_iter;
for (m_iter = members_in_id_order_.begin(); m_iter != members_in_id_order_.end(); ++m_iter) {
validate_union_member( *m_iter);
}
members_validated = true;
}
}
void set_union(bool is_union) {
is_union_ = is_union;
validate_union_members();
}
void set_xsd_all(bool xsd_all) {
xsd_all_ = xsd_all;
}
bool get_xsd_all() const {
return xsd_all_;
}
bool append(t_field* elem) {
typedef members_type::iterator iter_type;
std::pair<iter_type, iter_type> bounds = std::equal_range(
members_in_id_order_.begin(), members_in_id_order_.end(), elem, t_field::key_compare()
);
if (bounds.first != bounds.second) {
return false;
}
// returns false when there is a conflict of field names
if (get_field_by_name(elem->get_name()) != NULL) {
return false;
}
members_.push_back(elem);
members_in_id_order_.insert(bounds.second, elem);
validate_union_member( elem);
return true;
}
const members_type& get_members() {
return members_;
}
const members_type& get_sorted_members() {
return members_in_id_order_;
}
bool is_struct() const {
return !is_xception_;
}
bool is_xception() const {
return is_xception_;
}
bool is_union() const {
return is_union_;
}
virtual std::string get_fingerprint_material() const {
std::string rv = "{";
bool do_reserve = (members_in_id_order_.size() > 1);
size_t estimation = 0;
members_type::const_iterator m_iter;
for (m_iter = members_in_id_order_.begin(); m_iter != members_in_id_order_.end(); ++m_iter) {
rv += (*m_iter)->get_fingerprint_material();
rv += ";";
if( do_reserve) {
estimation = members_in_id_order_.size() * rv.size() + 16;
rv.reserve( estimation);
do_reserve = false;
}
}
rv += "}";
return rv;
}
virtual void generate_fingerprint() {
t_type::generate_fingerprint();
members_type::const_iterator m_iter;
for (m_iter = members_in_id_order_.begin(); m_iter != members_in_id_order_.end(); ++m_iter) {
(*m_iter)->get_type()->generate_fingerprint();
}
}
t_field* get_field_by_name(std::string field_name) {
members_type::const_iterator m_iter;
for (m_iter = members_in_id_order_.begin(); m_iter != members_in_id_order_.end(); ++m_iter) {
if ((*m_iter)->get_name() == field_name) {
return *m_iter;
}
}
return NULL;
}
private:
members_type members_;
members_type members_in_id_order_;
bool is_xception_;
bool is_union_;
bool members_validated;
int members_with_value;
bool xsd_all_;
};
#endif