/**
 * 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.12.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.12.0)")
public class SupervisorAssignments implements org.apache.storm.thrift.TBase<SupervisorAssignments, SupervisorAssignments._Fields>, java.io.Serializable, Cloneable, Comparable<SupervisorAssignments> {
  private static final org.apache.storm.thrift.protocol.TStruct STRUCT_DESC = new org.apache.storm.thrift.protocol.TStruct("SupervisorAssignments");

  private static final org.apache.storm.thrift.protocol.TField STORM_ASSIGNMENT_FIELD_DESC = new org.apache.storm.thrift.protocol.TField("storm_assignment", org.apache.storm.thrift.protocol.TType.MAP, (short)1);

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

  private @org.apache.storm.thrift.annotation.Nullable java.util.Map<java.lang.String,Assignment> storm_assignment; // 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 {
    STORM_ASSIGNMENT((short)1, "storm_assignment");

    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 1: // STORM_ASSIGNMENT
          return STORM_ASSIGNMENT;
        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.STORM_ASSIGNMENT};
  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.STORM_ASSIGNMENT, new org.apache.storm.thrift.meta_data.FieldMetaData("storm_assignment", 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, Assignment.class))));
    metaDataMap = java.util.Collections.unmodifiableMap(tmpMap);
    org.apache.storm.thrift.meta_data.FieldMetaData.addStructMetaDataMap(SupervisorAssignments.class, metaDataMap);
  }

  public SupervisorAssignments() {
    this.storm_assignment = new java.util.HashMap<java.lang.String,Assignment>();

  }

  /**
   * Performs a deep copy on <i>other</i>.
   */
  public SupervisorAssignments(SupervisorAssignments other) {
    if (other.is_set_storm_assignment()) {
      java.util.Map<java.lang.String,Assignment> __this__storm_assignment = new java.util.HashMap<java.lang.String,Assignment>(other.storm_assignment.size());
      for (java.util.Map.Entry<java.lang.String, Assignment> other_element : other.storm_assignment.entrySet()) {

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

        java.lang.String __this__storm_assignment_copy_key = other_element_key;

        Assignment __this__storm_assignment_copy_value = new Assignment(other_element_value);

        __this__storm_assignment.put(__this__storm_assignment_copy_key, __this__storm_assignment_copy_value);
      }
      this.storm_assignment = __this__storm_assignment;
    }
  }

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

  @Override
  public void clear() {
    this.storm_assignment = new java.util.HashMap<java.lang.String,Assignment>();

  }

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

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

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

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

  public void unset_storm_assignment() {
    this.storm_assignment = null;
  }

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

  public void set_storm_assignment_isSet(boolean value) {
    if (!value) {
      this.storm_assignment = null;
    }
  }

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

    }
  }

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

    }
    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 STORM_ASSIGNMENT:
      return is_set_storm_assignment();
    }
    throw new java.lang.IllegalStateException();
  }

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

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

    boolean this_present_storm_assignment = true && this.is_set_storm_assignment();
    boolean that_present_storm_assignment = true && that.is_set_storm_assignment();
    if (this_present_storm_assignment || that_present_storm_assignment) {
      if (!(this_present_storm_assignment && that_present_storm_assignment))
        return false;
      if (!this.storm_assignment.equals(that.storm_assignment))
        return false;
    }

    return true;
  }

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

    hashCode = hashCode * 8191 + ((is_set_storm_assignment()) ? 131071 : 524287);
    if (is_set_storm_assignment())
      hashCode = hashCode * 8191 + storm_assignment.hashCode();

    return hashCode;
  }

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

    int lastComparison = 0;

    lastComparison = java.lang.Boolean.valueOf(is_set_storm_assignment()).compareTo(other.is_set_storm_assignment());
    if (lastComparison != 0) {
      return lastComparison;
    }
    if (is_set_storm_assignment()) {
      lastComparison = org.apache.storm.thrift.TBaseHelper.compareTo(this.storm_assignment, other.storm_assignment);
      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("SupervisorAssignments(");
    boolean first = true;

    if (is_set_storm_assignment()) {
      sb.append("storm_assignment:");
      if (this.storm_assignment == null) {
        sb.append("null");
      } else {
        sb.append(this.storm_assignment);
      }
      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 SupervisorAssignmentsStandardSchemeFactory implements org.apache.storm.thrift.scheme.SchemeFactory {
    public SupervisorAssignmentsStandardScheme getScheme() {
      return new SupervisorAssignmentsStandardScheme();
    }
  }

  private static class SupervisorAssignmentsStandardScheme extends org.apache.storm.thrift.scheme.StandardScheme<SupervisorAssignments> {

    public void read(org.apache.storm.thrift.protocol.TProtocol iprot, SupervisorAssignments 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 1: // STORM_ASSIGNMENT
            if (schemeField.type == org.apache.storm.thrift.protocol.TType.MAP) {
              {
                org.apache.storm.thrift.protocol.TMap _map936 = iprot.readMapBegin();
                struct.storm_assignment = new java.util.HashMap<java.lang.String,Assignment>(2*_map936.size);
                @org.apache.storm.thrift.annotation.Nullable java.lang.String _key937;
                @org.apache.storm.thrift.annotation.Nullable Assignment _val938;
                for (int _i939 = 0; _i939 < _map936.size; ++_i939)
                {
                  _key937 = iprot.readString();
                  _val938 = new Assignment();
                  _val938.read(iprot);
                  struct.storm_assignment.put(_key937, _val938);
                }
                iprot.readMapEnd();
              }
              struct.set_storm_assignment_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, SupervisorAssignments struct) throws org.apache.storm.thrift.TException {
      struct.validate();

      oprot.writeStructBegin(STRUCT_DESC);
      if (struct.storm_assignment != null) {
        if (struct.is_set_storm_assignment()) {
          oprot.writeFieldBegin(STORM_ASSIGNMENT_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.storm_assignment.size()));
            for (java.util.Map.Entry<java.lang.String, Assignment> _iter940 : struct.storm_assignment.entrySet())
            {
              oprot.writeString(_iter940.getKey());
              _iter940.getValue().write(oprot);
            }
            oprot.writeMapEnd();
          }
          oprot.writeFieldEnd();
        }
      }
      oprot.writeFieldStop();
      oprot.writeStructEnd();
    }

  }

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

  private static class SupervisorAssignmentsTupleScheme extends org.apache.storm.thrift.scheme.TupleScheme<SupervisorAssignments> {

    @Override
    public void write(org.apache.storm.thrift.protocol.TProtocol prot, SupervisorAssignments 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_storm_assignment()) {
        optionals.set(0);
      }
      oprot.writeBitSet(optionals, 1);
      if (struct.is_set_storm_assignment()) {
        {
          oprot.writeI32(struct.storm_assignment.size());
          for (java.util.Map.Entry<java.lang.String, Assignment> _iter941 : struct.storm_assignment.entrySet())
          {
            oprot.writeString(_iter941.getKey());
            _iter941.getValue().write(oprot);
          }
        }
      }
    }

    @Override
    public void read(org.apache.storm.thrift.protocol.TProtocol prot, SupervisorAssignments 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 _map942 = new org.apache.storm.thrift.protocol.TMap(org.apache.storm.thrift.protocol.TType.STRING, org.apache.storm.thrift.protocol.TType.STRUCT, iprot.readI32());
          struct.storm_assignment = new java.util.HashMap<java.lang.String,Assignment>(2*_map942.size);
          @org.apache.storm.thrift.annotation.Nullable java.lang.String _key943;
          @org.apache.storm.thrift.annotation.Nullable Assignment _val944;
          for (int _i945 = 0; _i945 < _map942.size; ++_i945)
          {
            _key943 = iprot.readString();
            _val944 = new Assignment();
            _val944.read(iprot);
            struct.storm_assignment.put(_key943, _val944);
          }
        }
        struct.set_storm_assignment_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();
  }
}

