/*
 * Copyright 2016-present Open Networking Laboratory
 *
 * Licensed 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.
 */
package org.onosproject.isis.io.isispacket.tlv;

import com.google.common.base.MoreObjects;
import com.google.common.primitives.Bytes;
import org.jboss.netty.buffer.ChannelBuffer;
import org.onlab.packet.MacAddress;
import org.onosproject.isis.io.util.IsisUtil;

import java.util.ArrayList;
import java.util.List;

/**
 * Representation of  ISIS neighbor TLV.
 */
public class IsisNeighborTlv extends TlvHeader implements IsisTlv {

    private List<MacAddress> neighbor = new ArrayList<>();

    /**
     * Creates an instance of ISIS neighbor TLV.
     *
     * @param tlvHeader tlvHeader
     */
    public IsisNeighborTlv(TlvHeader tlvHeader) {
        this.setTlvType(tlvHeader.tlvType());
        this.setTlvLength(tlvHeader.tlvLength());
    }

    /**
     * Adds the MAC address of the neighbor to ISIS neighbor TLV.
     *
     * @param macAddress MAC address
     */
    public void addNeighbor(MacAddress macAddress) {
        neighbor.add(macAddress);
    }

    /**
     * Returns the MAC address of the ISIS neighbor TLV.
     *
     * @return neighbor
     */
    public List<MacAddress> neighbor() {
        return this.neighbor;
    }

    @Override
    public void readFrom(ChannelBuffer channelBuffer) {
        while (channelBuffer.readableBytes() >= 6) {
            byte[] addressbytes = new byte[IsisUtil.SIX_BYTES];
            channelBuffer.readBytes(addressbytes, 0, IsisUtil.SIX_BYTES);
            this.neighbor.add(MacAddress.valueOf(addressbytes));
        }
    }

    @Override
    public byte[] asBytes() {
        byte[] bytes = null;
        byte[] tlvHeader = tlvHeaderAsByteArray();
        byte[] tlvBody = tlvBodyAsBytes();
        tlvHeader[1] = (byte) tlvBody.length;
        bytes = Bytes.concat(tlvHeader, tlvBody);
        return bytes;
    }

    /**
     * Returns TLV body of ISIS neighbor TLV.
     *
     * @return byteArray TLV body of area address TLV
     */
    private byte[] tlvBodyAsBytes() {
        List<Byte> bytes = new ArrayList<>();
        for (MacAddress macAddress : this.neighbor) {
            bytes.addAll(Bytes.asList(macAddress.toBytes()));
        }
        return Bytes.toArray(bytes);
    }

    @Override
    public String toString() {
        return MoreObjects.toStringHelper(getClass())
                .omitNullValues()
                .add("neighbor", neighbor)
                .toString();
    }
}