/*
 * Copyright 2013 Anton Tananaev (anton.tananaev@gmail.com)
 *
 * 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.traccar.protocol;

import java.nio.ByteOrder;
import java.util.Date;
import org.jboss.netty.buffer.ChannelBuffer;
import org.jboss.netty.buffer.ChannelBuffers;
import org.jboss.netty.channel.Channel;
import org.jboss.netty.channel.ChannelHandlerContext;
import org.traccar.BaseProtocolDecoder;
import org.traccar.ServerManager;
import org.traccar.helper.Crc;
import org.traccar.helper.Log;
import org.traccar.model.ExtendedInfoFormatter;
import org.traccar.model.Position;

public class NavigilProtocolDecoder extends BaseProtocolDecoder {

    public NavigilProtocolDecoder(ServerManager serverManager) {
        super(serverManager);
    }
    
    private static final int LEAP_SECONDS_DELTA = 25;

    private static final int MESSAGE_ERROR = 2;
    private static final int MESSAGE_INDICATION = 4;
    private static final int MESSAGE_CONN_OPEN = 5;
    private static final int MESSAGE_CONN_CLOSE = 6;
    private static final int MESSAGE_SYSTEM_REPORT = 7;
    private static final int MESSAGE_UNIT_REPORT = 8;
    private static final int MESSAGE_GEOFENCE_ALARM = 10;
    private static final int MESSAGE_INPUT_ALARM = 11;
    private static final int MESSAGE_TG2_REPORT = 12;
    private static final int MESSAGE_POSITION_REPORT = 13;
    private static final int MESSAGE_POSITION_REPORT_2 = 15;
    private static final int MESSAGE_SNAPSHOT4 = 17;
    private static final int MESSAGE_TRACKING_DATA = 18;
    private static final int MESSAGE_MOTION_ALARM = 19;
    private static final int MESSAGE_ACKNOWLEDGEMENT = 255;
    
    private static Date convertTimestamp(long timestamp) {
        return new Date((timestamp - LEAP_SECONDS_DELTA) * 1000l);
    }
    
    private int senderSequenceNumber = 1;
    
    private void sendAcknowledgment(Channel channel, int sequenceNumber) {
        ChannelBuffer data = ChannelBuffers.directBuffer(ByteOrder.LITTLE_ENDIAN, 4);
        data.writeShort(sequenceNumber);
        data.writeShort(0); // OK
        
        ChannelBuffer header = ChannelBuffers.directBuffer(ByteOrder.LITTLE_ENDIAN, 20);
        header.writeByte(1); header.writeByte(0);
        header.writeShort(senderSequenceNumber++);
        header.writeShort(MESSAGE_ACKNOWLEDGEMENT);
        header.writeShort(header.capacity() + data.capacity());
        header.writeShort(0);
        header.writeShort(Crc.crc16X25Ccitt(data.toByteBuffer()));
        header.writeInt(0);
        header.writeInt((int) (new Date().getTime() / 1000) + LEAP_SECONDS_DELTA);
        
        if (channel != null) {
            channel.write(ChannelBuffers.copiedBuffer(header, data));
        }
    }
    
    private Position parseUnitReport(ChannelBuffer buf, long deviceId, int sequenceNumber) {
        Position position = new Position();
        ExtendedInfoFormatter extendedInfo = new ExtendedInfoFormatter("navigil");

        position.setValid(true);
        extendedInfo.set("index", sequenceNumber);
        position.setDeviceId(deviceId);
        
        buf.readUnsignedShort(); // report trigger
        buf.readUnsignedShort(); // flags
        
        position.setLatitude(buf.readInt() * 0.0000001);
        position.setLongitude(buf.readInt() * 0.0000001);
        position.setAltitude((double) buf.readUnsignedShort());
        
        buf.readUnsignedShort(); // satellites in fix
        buf.readUnsignedShort(); // satellites in track
        buf.readUnsignedShort(); // GPS antenna state
        
        position.setSpeed(buf.readUnsignedShort() * 0.194384);
        position.setCourse((double) buf.readUnsignedShort());
        
        buf.readUnsignedInt(); // distance
        buf.readUnsignedInt(); // delta distance

        extendedInfo.set("battery", buf.readUnsignedShort() * 0.001);
        
        buf.readUnsignedShort(); // battery charger status
        
        position.setTime(convertTimestamp(buf.readUnsignedInt()));
        
        // TODO: a lot of other stuff

        position.setExtendedInfo(extendedInfo.toString());
        return position;
    }
    
    private Position parseTg2Report(ChannelBuffer buf, long deviceId, int sequenceNumber) {
        Position position = new Position();
        ExtendedInfoFormatter extendedInfo = new ExtendedInfoFormatter("navigil");

        position.setValid(true);
        extendedInfo.set("index", sequenceNumber);
        position.setDeviceId(deviceId);
        
        buf.readUnsignedShort(); // report trigger
        buf.readUnsignedByte(); // reserved
        buf.readUnsignedByte(); // assisted GPS age
        
        position.setTime(convertTimestamp(buf.readUnsignedInt()));
        
        position.setLatitude(buf.readInt() * 0.0000001);
        position.setLongitude(buf.readInt() * 0.0000001);
        position.setAltitude((double) buf.readUnsignedShort());
        
        buf.readUnsignedByte(); // satellites in fix
        buf.readUnsignedByte(); // satellites in track
        
        position.setSpeed(buf.readUnsignedShort() * 0.194384);
        position.setCourse((double) buf.readUnsignedShort());
        
        buf.readUnsignedInt(); // distance
        buf.readUnsignedShort(); // maximum speed
        buf.readUnsignedShort(); // minimum speed

        buf.readUnsignedShort(); // VSAUT1 voltage
        buf.readUnsignedShort(); // VSAUT2 voltage
        buf.readUnsignedShort(); // solar voltage
        extendedInfo.set("battery", buf.readUnsignedShort() * 0.001);
        
        // TODO: a lot of other stuff

        position.setExtendedInfo(extendedInfo.toString());
        return position;
    }
    
    private Position parsePositionReport(ChannelBuffer buf, long deviceId, int sequenceNumber, long timestamp) {
        Position position = new Position();
        ExtendedInfoFormatter extendedInfo = new ExtendedInfoFormatter("navigil");

        extendedInfo.set("index", sequenceNumber);
        position.setDeviceId(deviceId);
        position.setTime(convertTimestamp(timestamp));
        
        position.setLatitude(buf.readMedium() * 0.00002);
        position.setLongitude(buf.readMedium() * 0.00002);
        position.setAltitude(0.0);
        
        position.setSpeed(buf.readUnsignedByte() * 0.539957);
        position.setCourse(buf.readUnsignedByte() * 2.0);
        
        short flags = buf.readUnsignedByte();
        position.setValid((flags & 0x80) == 0x80 && (flags & 0x40) == 0x40);
        
        buf.readUnsignedByte(); // reserved

        position.setExtendedInfo(extendedInfo.toString());
        return position;
    }
    
    private Position parsePositionReport2(ChannelBuffer buf, long deviceId, int sequenceNumber, long timestamp) {
        Position position = new Position();
        ExtendedInfoFormatter extendedInfo = new ExtendedInfoFormatter("navigil");

        extendedInfo.set("index", sequenceNumber);
        position.setDeviceId(deviceId);
        position.setTime(convertTimestamp(timestamp));
        
        position.setLatitude(buf.readInt() * 0.0000001);
        position.setLongitude(buf.readInt() * 0.0000001);
        position.setAltitude(0.0);
        
        buf.readUnsignedByte(); // report trigger

        position.setSpeed(buf.readUnsignedByte() * 0.539957);
        position.setCourse(0.0);
        
        short flags = buf.readUnsignedByte();
        position.setValid((flags & 0x80) == 0x80 && (flags & 0x40) == 0x40);
        
        int x = buf.readUnsignedByte(); // satellites in fix
        buf.readUnsignedInt(); // distance

        position.setExtendedInfo(extendedInfo.toString());
        return position;
    }
    
    private Position parseSnapshot4(ChannelBuffer buf, long deviceId, int sequenceNumber) {
        Position position = new Position();
        ExtendedInfoFormatter extendedInfo = new ExtendedInfoFormatter("navigil");

        extendedInfo.set("index", sequenceNumber);
        position.setDeviceId(deviceId);

        buf.readUnsignedByte(); // report trigger
        buf.readUnsignedByte(); // position fix source
        buf.readUnsignedByte(); // GNSS fix quality
        buf.readUnsignedByte(); // GNSS assistance age
        
        long flags = buf.readUnsignedInt();
        position.setValid((flags & 0x0400) == 0x0400);
        
        position.setTime(convertTimestamp(buf.readUnsignedInt()));
        
        position.setLatitude(buf.readInt() * 0.0000001);
        position.setLongitude(buf.readInt() * 0.0000001);
        position.setAltitude((double) buf.readUnsignedShort());
        
        buf.readUnsignedByte(); // satellites in fix
        buf.readUnsignedByte(); // satellites in track
        
        position.setSpeed(buf.readUnsignedShort() * 0.194384);
        position.setCourse(buf.readUnsignedShort() * 0.1);
        
        buf.readUnsignedByte(); // maximum speed
        buf.readUnsignedByte(); // minimum speed
        buf.readUnsignedInt(); // distance

        buf.readUnsignedByte(); // supply voltage 1
        buf.readUnsignedByte(); // supply voltage 2
        extendedInfo.set("battery", buf.readUnsignedShort() * 0.001);

        // TODO: a lot of other stuff

        position.setExtendedInfo(extendedInfo.toString());
        return position;
    }
    
    private Position parseTrackingData(ChannelBuffer buf, long deviceId, int sequenceNumber, long timestamp) {
        Position position = new Position();
        ExtendedInfoFormatter extendedInfo = new ExtendedInfoFormatter("navigil");

        extendedInfo.set("index", sequenceNumber);
        position.setDeviceId(deviceId);
        position.setTime(convertTimestamp(timestamp));

        buf.readUnsignedByte(); // tracking mode
        
        short flags = buf.readUnsignedByte();
        position.setValid((flags & 0x01) == 0x01);
        
        buf.readUnsignedShort(); // duration

        position.setLatitude(buf.readInt() * 0.0000001);
        position.setLongitude(buf.readInt() * 0.0000001);
        position.setAltitude(0.0);
        
        position.setSpeed(buf.readUnsignedByte() * 0.539957);
        position.setCourse(buf.readUnsignedByte() * 2.0);

        buf.readUnsignedByte(); // satellites in fix
        
        extendedInfo.set("battery", buf.readUnsignedShort() * 0.001);
        
        buf.readUnsignedInt(); // distance

        position.setExtendedInfo(extendedInfo.toString());
        return position;
    }
    
    @Override
    protected Object decode(ChannelHandlerContext ctx, Channel channel, Object msg)
            throws Exception {
        
        ChannelBuffer buf = (ChannelBuffer) msg;
        
        buf.readUnsignedByte(); // protocol version
        buf.readUnsignedByte(); // version id
        int sequenceNumber = buf.readUnsignedShort();
        int messageId = buf.readUnsignedShort();
        buf.readUnsignedShort(); // length
        int flags = buf.readUnsignedShort();
        buf.readUnsignedShort(); // checksum
        
        // Get device identifier
        long deviceId;
        String navigilDeviceId = String.valueOf(buf.readUnsignedInt());
        try {
            deviceId = getDataManager().getDeviceByImei(navigilDeviceId).getId();
        } catch(Exception error) {
            Log.warning("Unknown device - " + navigilDeviceId);
            return null;
        }

        long timestamp = buf.readUnsignedInt(); // message timestamp

        // Acknowledgment
        if ((flags & 0x1) == 0x0) {
            sendAcknowledgment(channel, sequenceNumber);
        }
        
        // Parse messages
        switch (messageId) {
            case MESSAGE_UNIT_REPORT:
                return parseUnitReport(buf, deviceId, sequenceNumber);
            case MESSAGE_TG2_REPORT:
                return parseTg2Report(buf, deviceId, sequenceNumber);
            case MESSAGE_POSITION_REPORT:
                return parsePositionReport(buf, deviceId, sequenceNumber, timestamp);
            case MESSAGE_POSITION_REPORT_2:
                return parsePositionReport2(buf, deviceId, sequenceNumber, timestamp);
            case MESSAGE_SNAPSHOT4:
                return parseSnapshot4(buf, deviceId, sequenceNumber);
            case MESSAGE_TRACKING_DATA:
                return parseTrackingData(buf, deviceId, sequenceNumber, timestamp);
        }
        
        return null;
    }

}