/**
 * 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.
 */
/**
 * Autogenerated by Thrift Compiler (0.13.0)
 *
 * DO NOT EDIT UNLESS YOU ARE SURE THAT YOU KNOW WHAT YOU ARE DOING
 *  @generated
 */
package org.apache.storm.generated;

@SuppressWarnings({"cast", "rawtypes", "serial", "unchecked", "unused"})
@javax.annotation.Generated(value = "Autogenerated by Thrift Compiler (0.13.0)")
public class LogConfig implements org.apache.storm.thrift.TBase<LogConfig, LogConfig._Fields>, java.io.Serializable, Cloneable, Comparable<LogConfig> {
  private static final org.apache.storm.thrift.protocol.TStruct STRUCT_DESC = new org.apache.storm.thrift.protocol.TStruct("LogConfig");

  private static final org.apache.storm.thrift.protocol.TField NAMED_LOGGER_LEVEL_FIELD_DESC = new org.apache.storm.thrift.protocol.TField("named_logger_level", org.apache.storm.thrift.protocol.TType.MAP, (short)2);

  private static final org.apache.storm.thrift.scheme.SchemeFactory STANDARD_SCHEME_FACTORY = new LogConfigStandardSchemeFactory();
  private static final org.apache.storm.thrift.scheme.SchemeFactory TUPLE_SCHEME_FACTORY = new LogConfigTupleSchemeFactory();

  private @org.apache.storm.thrift.annotation.Nullable java.util.Map<java.lang.String,LogLevel> named_logger_level; // optional

  /** The set of fields this struct contains, along with convenience methods for finding and manipulating them. */
  public enum _Fields implements org.apache.storm.thrift.TFieldIdEnum {
    NAMED_LOGGER_LEVEL((short)2, "named_logger_level");

    private static final java.util.Map<java.lang.String, _Fields> byName = new java.util.HashMap<java.lang.String, _Fields>();

    static {
      for (_Fields field : java.util.EnumSet.allOf(_Fields.class)) {
        byName.put(field.getFieldName(), field);
      }
    }

    /**
     * Find the _Fields constant that matches fieldId, or null if its not found.
     */
    @org.apache.storm.thrift.annotation.Nullable
    public static _Fields findByThriftId(int fieldId) {
      switch(fieldId) {
        case 2: // NAMED_LOGGER_LEVEL
          return NAMED_LOGGER_LEVEL;
        default:
          return null;
      }
    }

    /**
     * Find the _Fields constant that matches fieldId, throwing an exception
     * if it is not found.
     */
    public static _Fields findByThriftIdOrThrow(int fieldId) {
      _Fields fields = findByThriftId(fieldId);
      if (fields == null) throw new java.lang.IllegalArgumentException("Field " + fieldId + " doesn't exist!");
      return fields;
    }

    /**
     * Find the _Fields constant that matches name, or null if its not found.
     */
    @org.apache.storm.thrift.annotation.Nullable
    public static _Fields findByName(java.lang.String name) {
      return byName.get(name);
    }

    private final short _thriftId;
    private final java.lang.String _fieldName;

    _Fields(short thriftId, java.lang.String fieldName) {
      _thriftId = thriftId;
      _fieldName = fieldName;
    }

    public short getThriftFieldId() {
      return _thriftId;
    }

    public java.lang.String getFieldName() {
      return _fieldName;
    }
  }

  // isset id assignments
  private static final _Fields optionals[] = {_Fields.NAMED_LOGGER_LEVEL};
  public static final java.util.Map<_Fields, org.apache.storm.thrift.meta_data.FieldMetaData> metaDataMap;
  static {
    java.util.Map<_Fields, org.apache.storm.thrift.meta_data.FieldMetaData> tmpMap = new java.util.EnumMap<_Fields, org.apache.storm.thrift.meta_data.FieldMetaData>(_Fields.class);
    tmpMap.put(_Fields.NAMED_LOGGER_LEVEL, new org.apache.storm.thrift.meta_data.FieldMetaData("named_logger_level", org.apache.storm.thrift.TFieldRequirementType.OPTIONAL, 
        new org.apache.storm.thrift.meta_data.MapMetaData(org.apache.storm.thrift.protocol.TType.MAP, 
            new org.apache.storm.thrift.meta_data.FieldValueMetaData(org.apache.storm.thrift.protocol.TType.STRING), 
            new org.apache.storm.thrift.meta_data.StructMetaData(org.apache.storm.thrift.protocol.TType.STRUCT, LogLevel.class))));
    metaDataMap = java.util.Collections.unmodifiableMap(tmpMap);
    org.apache.storm.thrift.meta_data.FieldMetaData.addStructMetaDataMap(LogConfig.class, metaDataMap);
  }

  public LogConfig() {
  }

  /**
   * Performs a deep copy on <i>other</i>.
   */
  public LogConfig(LogConfig other) {
    if (other.is_set_named_logger_level()) {
      java.util.Map<java.lang.String,LogLevel> __this__named_logger_level = new java.util.HashMap<java.lang.String,LogLevel>(other.named_logger_level.size());
      for (java.util.Map.Entry<java.lang.String, LogLevel> other_element : other.named_logger_level.entrySet()) {

        java.lang.String other_element_key = other_element.getKey();
        LogLevel other_element_value = other_element.getValue();

        java.lang.String __this__named_logger_level_copy_key = other_element_key;

        LogLevel __this__named_logger_level_copy_value = new LogLevel(other_element_value);

        __this__named_logger_level.put(__this__named_logger_level_copy_key, __this__named_logger_level_copy_value);
      }
      this.named_logger_level = __this__named_logger_level;
    }
  }

  public LogConfig deepCopy() {
    return new LogConfig(this);
  }

  @Override
  public void clear() {
    this.named_logger_level = null;
  }

  public int get_named_logger_level_size() {
    return (this.named_logger_level == null) ? 0 : this.named_logger_level.size();
  }

  public void put_to_named_logger_level(java.lang.String key, LogLevel val) {
    if (this.named_logger_level == null) {
      this.named_logger_level = new java.util.HashMap<java.lang.String,LogLevel>();
    }
    this.named_logger_level.put(key, val);
  }

  @org.apache.storm.thrift.annotation.Nullable
  public java.util.Map<java.lang.String,LogLevel> get_named_logger_level() {
    return this.named_logger_level;
  }

  public void set_named_logger_level(@org.apache.storm.thrift.annotation.Nullable java.util.Map<java.lang.String,LogLevel> named_logger_level) {
    this.named_logger_level = named_logger_level;
  }

  public void unset_named_logger_level() {
    this.named_logger_level = null;
  }

  /** Returns true if field named_logger_level is set (has been assigned a value) and false otherwise */
  public boolean is_set_named_logger_level() {
    return this.named_logger_level != null;
  }

  public void set_named_logger_level_isSet(boolean value) {
    if (!value) {
      this.named_logger_level = null;
    }
  }

  public void setFieldValue(_Fields field, @org.apache.storm.thrift.annotation.Nullable java.lang.Object value) {
    switch (field) {
    case NAMED_LOGGER_LEVEL:
      if (value == null) {
        unset_named_logger_level();
      } else {
        set_named_logger_level((java.util.Map<java.lang.String,LogLevel>)value);
      }
      break;

    }
  }

  @org.apache.storm.thrift.annotation.Nullable
  public java.lang.Object getFieldValue(_Fields field) {
    switch (field) {
    case NAMED_LOGGER_LEVEL:
      return get_named_logger_level();

    }
    throw new java.lang.IllegalStateException();
  }

  /** Returns true if field corresponding to fieldID is set (has been assigned a value) and false otherwise */
  public boolean isSet(_Fields field) {
    if (field == null) {
      throw new java.lang.IllegalArgumentException();
    }

    switch (field) {
    case NAMED_LOGGER_LEVEL:
      return is_set_named_logger_level();
    }
    throw new java.lang.IllegalStateException();
  }

  @Override
  public boolean equals(java.lang.Object that) {
    if (that == null)
      return false;
    if (that instanceof LogConfig)
      return this.equals((LogConfig)that);
    return false;
  }

  public boolean equals(LogConfig that) {
    if (that == null)
      return false;
    if (this == that)
      return true;

    boolean this_present_named_logger_level = true && this.is_set_named_logger_level();
    boolean that_present_named_logger_level = true && that.is_set_named_logger_level();
    if (this_present_named_logger_level || that_present_named_logger_level) {
      if (!(this_present_named_logger_level && that_present_named_logger_level))
        return false;
      if (!this.named_logger_level.equals(that.named_logger_level))
        return false;
    }

    return true;
  }

  @Override
  public int hashCode() {
    int hashCode = 1;

    hashCode = hashCode * 8191 + ((is_set_named_logger_level()) ? 131071 : 524287);
    if (is_set_named_logger_level())
      hashCode = hashCode * 8191 + named_logger_level.hashCode();

    return hashCode;
  }

  @Override
  public int compareTo(LogConfig other) {
    if (!getClass().equals(other.getClass())) {
      return getClass().getName().compareTo(other.getClass().getName());
    }

    int lastComparison = 0;

    lastComparison = java.lang.Boolean.valueOf(is_set_named_logger_level()).compareTo(other.is_set_named_logger_level());
    if (lastComparison != 0) {
      return lastComparison;
    }
    if (is_set_named_logger_level()) {
      lastComparison = org.apache.storm.thrift.TBaseHelper.compareTo(this.named_logger_level, other.named_logger_level);
      if (lastComparison != 0) {
        return lastComparison;
      }
    }
    return 0;
  }

  @org.apache.storm.thrift.annotation.Nullable
  public _Fields fieldForId(int fieldId) {
    return _Fields.findByThriftId(fieldId);
  }

  public void read(org.apache.storm.thrift.protocol.TProtocol iprot) throws org.apache.storm.thrift.TException {
    scheme(iprot).read(iprot, this);
  }

  public void write(org.apache.storm.thrift.protocol.TProtocol oprot) throws org.apache.storm.thrift.TException {
    scheme(oprot).write(oprot, this);
  }

  @Override
  public java.lang.String toString() {
    java.lang.StringBuilder sb = new java.lang.StringBuilder("LogConfig(");
    boolean first = true;

    if (is_set_named_logger_level()) {
      sb.append("named_logger_level:");
      if (this.named_logger_level == null) {
        sb.append("null");
      } else {
        sb.append(this.named_logger_level);
      }
      first = false;
    }
    sb.append(")");
    return sb.toString();
  }

  public void validate() throws org.apache.storm.thrift.TException {
    // check for required fields
    // check for sub-struct validity
  }

  private void writeObject(java.io.ObjectOutputStream out) throws java.io.IOException {
    try {
      write(new org.apache.storm.thrift.protocol.TCompactProtocol(new org.apache.storm.thrift.transport.TIOStreamTransport(out)));
    } catch (org.apache.storm.thrift.TException te) {
      throw new java.io.IOException(te);
    }
  }

  private void readObject(java.io.ObjectInputStream in) throws java.io.IOException, java.lang.ClassNotFoundException {
    try {
      read(new org.apache.storm.thrift.protocol.TCompactProtocol(new org.apache.storm.thrift.transport.TIOStreamTransport(in)));
    } catch (org.apache.storm.thrift.TException te) {
      throw new java.io.IOException(te);
    }
  }

  private static class LogConfigStandardSchemeFactory implements org.apache.storm.thrift.scheme.SchemeFactory {
    public LogConfigStandardScheme getScheme() {
      return new LogConfigStandardScheme();
    }
  }

  private static class LogConfigStandardScheme extends org.apache.storm.thrift.scheme.StandardScheme<LogConfig> {

    public void read(org.apache.storm.thrift.protocol.TProtocol iprot, LogConfig struct) throws org.apache.storm.thrift.TException {
      org.apache.storm.thrift.protocol.TField schemeField;
      iprot.readStructBegin();
      while (true)
      {
        schemeField = iprot.readFieldBegin();
        if (schemeField.type == org.apache.storm.thrift.protocol.TType.STOP) { 
          break;
        }
        switch (schemeField.id) {
          case 2: // NAMED_LOGGER_LEVEL
            if (schemeField.type == org.apache.storm.thrift.protocol.TType.MAP) {
              {
                org.apache.storm.thrift.protocol.TMap _map902 = iprot.readMapBegin();
                struct.named_logger_level = new java.util.HashMap<java.lang.String,LogLevel>(2*_map902.size);
                @org.apache.storm.thrift.annotation.Nullable java.lang.String _key903;
                @org.apache.storm.thrift.annotation.Nullable LogLevel _val904;
                for (int _i905 = 0; _i905 < _map902.size; ++_i905)
                {
                  _key903 = iprot.readString();
                  _val904 = new LogLevel();
                  _val904.read(iprot);
                  struct.named_logger_level.put(_key903, _val904);
                }
                iprot.readMapEnd();
              }
              struct.set_named_logger_level_isSet(true);
            } else { 
              org.apache.storm.thrift.protocol.TProtocolUtil.skip(iprot, schemeField.type);
            }
            break;
          default:
            org.apache.storm.thrift.protocol.TProtocolUtil.skip(iprot, schemeField.type);
        }
        iprot.readFieldEnd();
      }
      iprot.readStructEnd();
      struct.validate();
    }

    public void write(org.apache.storm.thrift.protocol.TProtocol oprot, LogConfig struct) throws org.apache.storm.thrift.TException {
      struct.validate();

      oprot.writeStructBegin(STRUCT_DESC);
      if (struct.named_logger_level != null) {
        if (struct.is_set_named_logger_level()) {
          oprot.writeFieldBegin(NAMED_LOGGER_LEVEL_FIELD_DESC);
          {
            oprot.writeMapBegin(new org.apache.storm.thrift.protocol.TMap(org.apache.storm.thrift.protocol.TType.STRING, org.apache.storm.thrift.protocol.TType.STRUCT, struct.named_logger_level.size()));
            for (java.util.Map.Entry<java.lang.String, LogLevel> _iter906 : struct.named_logger_level.entrySet())
            {
              oprot.writeString(_iter906.getKey());
              _iter906.getValue().write(oprot);
            }
            oprot.writeMapEnd();
          }
          oprot.writeFieldEnd();
        }
      }
      oprot.writeFieldStop();
      oprot.writeStructEnd();
    }

  }

  private static class LogConfigTupleSchemeFactory implements org.apache.storm.thrift.scheme.SchemeFactory {
    public LogConfigTupleScheme getScheme() {
      return new LogConfigTupleScheme();
    }
  }

  private static class LogConfigTupleScheme extends org.apache.storm.thrift.scheme.TupleScheme<LogConfig> {

    @Override
    public void write(org.apache.storm.thrift.protocol.TProtocol prot, LogConfig struct) throws org.apache.storm.thrift.TException {
      org.apache.storm.thrift.protocol.TTupleProtocol oprot = (org.apache.storm.thrift.protocol.TTupleProtocol) prot;
      java.util.BitSet optionals = new java.util.BitSet();
      if (struct.is_set_named_logger_level()) {
        optionals.set(0);
      }
      oprot.writeBitSet(optionals, 1);
      if (struct.is_set_named_logger_level()) {
        {
          oprot.writeI32(struct.named_logger_level.size());
          for (java.util.Map.Entry<java.lang.String, LogLevel> _iter907 : struct.named_logger_level.entrySet())
          {
            oprot.writeString(_iter907.getKey());
            _iter907.getValue().write(oprot);
          }
        }
      }
    }

    @Override
    public void read(org.apache.storm.thrift.protocol.TProtocol prot, LogConfig struct) throws org.apache.storm.thrift.TException {
      org.apache.storm.thrift.protocol.TTupleProtocol iprot = (org.apache.storm.thrift.protocol.TTupleProtocol) prot;
      java.util.BitSet incoming = iprot.readBitSet(1);
      if (incoming.get(0)) {
        {
          org.apache.storm.thrift.protocol.TMap _map908 = new org.apache.storm.thrift.protocol.TMap(org.apache.storm.thrift.protocol.TType.STRING, org.apache.storm.thrift.protocol.TType.STRUCT, iprot.readI32());
          struct.named_logger_level = new java.util.HashMap<java.lang.String,LogLevel>(2*_map908.size);
          @org.apache.storm.thrift.annotation.Nullable java.lang.String _key909;
          @org.apache.storm.thrift.annotation.Nullable LogLevel _val910;
          for (int _i911 = 0; _i911 < _map908.size; ++_i911)
          {
            _key909 = iprot.readString();
            _val910 = new LogLevel();
            _val910.read(iprot);
            struct.named_logger_level.put(_key909, _val910);
          }
        }
        struct.set_named_logger_level_isSet(true);
      }
    }
  }

  private static <S extends org.apache.storm.thrift.scheme.IScheme> S scheme(org.apache.storm.thrift.protocol.TProtocol proto) {
    return (org.apache.storm.thrift.scheme.StandardScheme.class.equals(proto.getScheme()) ? STANDARD_SCHEME_FACTORY : TUPLE_SCHEME_FACTORY).getScheme();
  }
}

