aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAnton Tananaev <anton.tananaev@gmail.com>2015-05-31 20:07:39 +1200
committerAnton Tananaev <anton.tananaev@gmail.com>2015-05-31 20:07:39 +1200
commit57672030502b403fddd6a8f6b0a5dde35202c9c3 (patch)
treea3f7623a0035202acfe5ba9855ee3ae391104e26
parent70863c500e4cba47eb09b08848a2105ebb108e31 (diff)
downloadtraccar-server-57672030502b403fddd6a8f6b0a5dde35202c9c3.tar.gz
traccar-server-57672030502b403fddd6a8f6b0a5dde35202c9c3.tar.bz2
traccar-server-57672030502b403fddd6a8f6b0a5dde35202c9c3.zip
Implement Castel protocol
-rw-r--r--default.cfg4
-rw-r--r--src/org/traccar/ServerManager.java135
-rw-r--r--src/org/traccar/protocol/CastelProtocolDecoder.java141
-rw-r--r--test/org/traccar/protocol/CastelProtocolDecoderTest.java28
-rw-r--r--test/org/traccar/protocol/MeitrackProtocolDecoderTest.java4
5 files changed, 309 insertions, 3 deletions
diff --git a/default.cfg b/default.cfg
index bc552c9cb..6226f10f3 100644
--- a/default.cfg
+++ b/default.cfg
@@ -418,4 +418,8 @@
<entry key='avl301.enable'>true</entry>
<entry key='avl301.port'>5085</entry>
+ <!-- Castel server configuration -->
+ <entry key='castel.enable'>true</entry>
+ <entry key='castel.port'>5086</entry>
+
</properties>
diff --git a/src/org/traccar/ServerManager.java b/src/org/traccar/ServerManager.java
index 9e5da51b1..13ca885a9 100644
--- a/src/org/traccar/ServerManager.java
+++ b/src/org/traccar/ServerManager.java
@@ -19,7 +19,6 @@ import java.nio.ByteOrder;
import java.sql.SQLException;
import java.util.LinkedList;
import java.util.List;
-
import org.jboss.netty.bootstrap.ConnectionlessBootstrap;
import org.jboss.netty.bootstrap.ServerBootstrap;
import org.jboss.netty.channel.ChannelPipeline;
@@ -31,7 +30,114 @@ import org.jboss.netty.handler.codec.http.HttpRequestDecoder;
import org.jboss.netty.handler.codec.http.HttpResponseEncoder;
import org.jboss.netty.handler.codec.string.StringDecoder;
import org.jboss.netty.handler.codec.string.StringEncoder;
-import org.traccar.protocol.*;
+import org.traccar.protocol.ApelProtocolDecoder;
+import org.traccar.protocol.AplicomFrameDecoder;
+import org.traccar.protocol.AplicomProtocolDecoder;
+import org.traccar.protocol.Ardi01ProtocolDecoder;
+import org.traccar.protocol.AtrackFrameDecoder;
+import org.traccar.protocol.AtrackProtocolDecoder;
+import org.traccar.protocol.AutoFon45FrameDecoder;
+import org.traccar.protocol.AutoFon45ProtocolDecoder;
+import org.traccar.protocol.AutoFonFrameDecoder;
+import org.traccar.protocol.AutoFonProtocolDecoder;
+import org.traccar.protocol.Avl301ProtocolDecoder;
+import org.traccar.protocol.BceFrameDecoder;
+import org.traccar.protocol.BceProtocolDecoder;
+import org.traccar.protocol.BoxProtocolDecoder;
+import org.traccar.protocol.CalAmpProtocolDecoder;
+import org.traccar.protocol.CarTrackProtocolDecoder;
+import org.traccar.protocol.CarscopProtocolDecoder;
+import org.traccar.protocol.CastelProtocolDecoder;
+import org.traccar.protocol.CellocatorFrameDecoder;
+import org.traccar.protocol.CellocatorProtocolDecoder;
+import org.traccar.protocol.EasyTrackProtocolDecoder;
+import org.traccar.protocol.EelinkProtocolDecoder;
+import org.traccar.protocol.EnforaProtocolDecoder;
+import org.traccar.protocol.Ev603ProtocolDecoder;
+import org.traccar.protocol.FreedomProtocolDecoder;
+import org.traccar.protocol.GalileoFrameDecoder;
+import org.traccar.protocol.GalileoProtocolDecoder;
+import org.traccar.protocol.GatorProtocolDecoder;
+import org.traccar.protocol.Gl100ProtocolDecoder;
+import org.traccar.protocol.Gl200ProtocolDecoder;
+import org.traccar.protocol.GlobalSatProtocolDecoder;
+import org.traccar.protocol.GoSafeProtocolDecoder;
+import org.traccar.protocol.GotopProtocolDecoder;
+import org.traccar.protocol.Gps103ProtocolDecoder;
+import org.traccar.protocol.GpsGateProtocolDecoder;
+import org.traccar.protocol.Gt02ProtocolDecoder;
+import org.traccar.protocol.Gt06FrameDecoder;
+import org.traccar.protocol.Gt06ProtocolDecoder;
+import org.traccar.protocol.H02FrameDecoder;
+import org.traccar.protocol.H02ProtocolDecoder;
+import org.traccar.protocol.HaicomProtocolDecoder;
+import org.traccar.protocol.IntellitracFrameDecoder;
+import org.traccar.protocol.IntellitracProtocolDecoder;
+import org.traccar.protocol.Jt600FrameDecoder;
+import org.traccar.protocol.Jt600ProtocolDecoder;
+import org.traccar.protocol.KhdProtocolDecoder;
+import org.traccar.protocol.LaipacProtocolDecoder;
+import org.traccar.protocol.M2mProtocolDecoder;
+import org.traccar.protocol.ManPowerProtocolDecoder;
+import org.traccar.protocol.MaxonProtocolDecoder;
+import org.traccar.protocol.MegastekProtocolDecoder;
+import org.traccar.protocol.MeiligaoFrameDecoder;
+import org.traccar.protocol.MeiligaoProtocolDecoder;
+import org.traccar.protocol.MeitrackFrameDecoder;
+import org.traccar.protocol.MeitrackProtocolDecoder;
+import org.traccar.protocol.MiniFinderProtocolDecoder;
+import org.traccar.protocol.Mta6ProtocolDecoder;
+import org.traccar.protocol.MtxProtocolDecoder;
+import org.traccar.protocol.NavigilFrameDecoder;
+import org.traccar.protocol.NavigilProtocolDecoder;
+import org.traccar.protocol.NavisProtocolDecoder;
+import org.traccar.protocol.NoranProtocolDecoder;
+import org.traccar.protocol.OrionFrameDecoder;
+import org.traccar.protocol.OrionProtocolDecoder;
+import org.traccar.protocol.OsmAndProtocolDecoder;
+import org.traccar.protocol.PiligrimProtocolDecoder;
+import org.traccar.protocol.ProgressProtocolDecoder;
+import org.traccar.protocol.Pt3000ProtocolDecoder;
+import org.traccar.protocol.Pt502FrameDecoder;
+import org.traccar.protocol.Pt502ProtocolDecoder;
+import org.traccar.protocol.RitiProtocolDecoder;
+import org.traccar.protocol.RuptelaProtocolDecoder;
+import org.traccar.protocol.SanavProtocolDecoder;
+import org.traccar.protocol.SkypatrolProtocolDecoder;
+import org.traccar.protocol.Stl060FrameDecoder;
+import org.traccar.protocol.Stl060ProtocolDecoder;
+import org.traccar.protocol.SuntechProtocolDecoder;
+import org.traccar.protocol.SyrusProtocolDecoder;
+import org.traccar.protocol.T55ProtocolDecoder;
+import org.traccar.protocol.TelikProtocolDecoder;
+import org.traccar.protocol.TeltonikaFrameDecoder;
+import org.traccar.protocol.TeltonikaProtocolDecoder;
+import org.traccar.protocol.Tk102ProtocolDecoder;
+import org.traccar.protocol.Tk103ProtocolDecoder;
+import org.traccar.protocol.Tlt2hProtocolDecoder;
+import org.traccar.protocol.TopflytechProtocolDecoder;
+import org.traccar.protocol.TotemFrameDecoder;
+import org.traccar.protocol.TotemProtocolDecoder;
+import org.traccar.protocol.Tr20ProtocolDecoder;
+import org.traccar.protocol.Tr900ProtocolDecoder;
+import org.traccar.protocol.TrackboxProtocolDecoder;
+import org.traccar.protocol.TramigoFrameDecoder;
+import org.traccar.protocol.TramigoProtocolDecoder;
+import org.traccar.protocol.TytanProtocolDecoder;
+import org.traccar.protocol.UlbotechFrameDecoder;
+import org.traccar.protocol.UlbotechProtocolDecoder;
+import org.traccar.protocol.V680ProtocolDecoder;
+import org.traccar.protocol.VisiontekProtocolDecoder;
+import org.traccar.protocol.WialonProtocolDecoder;
+import org.traccar.protocol.WondexFrameDecoder;
+import org.traccar.protocol.WondexProtocolDecoder;
+import org.traccar.protocol.Xexun2ProtocolDecoder;
+import org.traccar.protocol.XexunFrameDecoder;
+import org.traccar.protocol.XexunProtocolDecoder;
+import org.traccar.protocol.XirgoProtocolDecoder;
+import org.traccar.protocol.Xt013ProtocolDecoder;
+import org.traccar.protocol.Xt7ProtocolDecoder;
+import org.traccar.protocol.YwtProtocolDecoder;
public class ServerManager {
@@ -125,6 +231,7 @@ public class ServerManager {
initMtxServer("mtx");
initTytanServer("tytan");
initAvl301Server("avl301");
+ initCastelServer("castel");
initProtocolDetector();
}
@@ -170,6 +277,7 @@ public class ServerManager {
});
}
}
+
private void initGps103Server(final String protocol) throws SQLException {
if (isProtocolEnabled(protocol)) {
serverList.add(new TrackerServer(new ServerBootstrap(), protocol) {
@@ -1326,4 +1434,27 @@ public class ServerManager {
}
}
+ private void initCastelServer(final String protocol) throws SQLException {
+ if (isProtocolEnabled(protocol)) {
+ TrackerServer server = new TrackerServer(new ServerBootstrap(), protocol) {
+ @Override
+ protected void addSpecificHandlers(ChannelPipeline pipeline) {
+ pipeline.addLast("frameDecoder", new LengthFieldBasedFrameDecoder(1024, 2, 2, -2, 0));
+ pipeline.addLast("objectDecoder", new CastelProtocolDecoder(protocol));
+ }
+ };
+ server.setEndianness(ByteOrder.LITTLE_ENDIAN);
+ serverList.add(server);
+
+ server = new TrackerServer(new ConnectionlessBootstrap(), protocol) {
+ @Override
+ protected void addSpecificHandlers(ChannelPipeline pipeline) {
+ pipeline.addLast("objectDecoder", new Gps103ProtocolDecoder(protocol));
+ }
+ };
+ server.setEndianness(ByteOrder.LITTLE_ENDIAN);
+ serverList.add(server);
+ }
+ }
+
}
diff --git a/src/org/traccar/protocol/CastelProtocolDecoder.java b/src/org/traccar/protocol/CastelProtocolDecoder.java
new file mode 100644
index 000000000..071218bf8
--- /dev/null
+++ b/src/org/traccar/protocol/CastelProtocolDecoder.java
@@ -0,0 +1,141 @@
+/*
+ * Copyright 2015 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.net.SocketAddress;
+import java.nio.charset.Charset;
+import java.util.Calendar;
+import java.util.TimeZone;
+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.helper.Crc;
+import org.traccar.helper.UnitsConverter;
+import org.traccar.model.Event;
+import org.traccar.model.Position;
+
+public class CastelProtocolDecoder extends BaseProtocolDecoder {
+
+ public CastelProtocolDecoder(String protocol) {
+ super(protocol);
+ }
+
+ private static final int MSG_LOGIN = 0x0110;
+ private static final int MSG_LOGIN_RESPONSE = 0x0190;
+ private static final int MSG_HEARTBEAT = 0x0301;
+ private static final int MSG_HEARTBEAT_RESPONSE = 0x0390;
+ private static final int MSG_GPS = 0x0140;
+
+ @Override
+ protected Object decode(
+ ChannelHandlerContext ctx, Channel channel, SocketAddress remoteAddress, Object msg)
+ throws Exception {
+
+ ChannelBuffer buf = (ChannelBuffer) msg;
+
+ buf.skipBytes(2); // header
+ buf.readUnsignedShort(); // length
+ int version = buf.readUnsignedByte();
+ ChannelBuffer id = buf.readBytes(20);
+ int type = buf.readUnsignedShort();
+
+ if (type == MSG_HEARTBEAT) {
+
+ if (channel != null) {
+ ChannelBuffer response = ChannelBuffers.directBuffer(31);
+ response.writeByte(0x40); response.writeByte(0x40); // header
+ response.writeShort(response.capacity()); // size
+ response.writeByte(version);
+ response.writeBytes(id);
+ response.writeShort(MSG_HEARTBEAT_RESPONSE);
+ response.writeShort(Crc.crc16Ccitt(response.toByteBuffer(2, 4))); // TODO
+ response.writeByte(0x0D); response.writeByte(0x0A); // ending
+ channel.write(response, remoteAddress);
+ }
+
+ } else if (type == MSG_LOGIN || type == MSG_GPS) {
+
+ Position position = new Position();
+ position.setDeviceId(getDeviceId());
+ position.setProtocol(getProtocol());
+
+ if (!identify(id.toString(Charset.defaultCharset()))) {
+ return null;
+ } else if (type == MSG_LOGIN) {
+
+ if (channel != null) {
+ ChannelBuffer response = ChannelBuffers.directBuffer(41);
+ response.writeByte(0x40); response.writeByte(0x40); // header
+ response.writeShort(response.capacity()); // size
+ response.writeByte(version);
+ response.writeBytes(id);
+ response.writeShort(MSG_LOGIN_RESPONSE);
+ response.writeInt(0xFFFFFFFF);
+ response.writeShort(0);
+ response.writeInt(0); // TODO
+ response.writeShort(Crc.crc16Ccitt(response.toByteBuffer(2, 4))); // TODO
+ response.writeByte(0x0D); response.writeByte(0x0A); // ending
+ channel.write(response, remoteAddress);
+ }
+
+ }
+
+ if (type == MSG_GPS) {
+ buf.readUnsignedByte(); // historical
+ }
+
+ buf.readUnsignedInt(); // ACC ON time
+ buf.readUnsignedInt(); // UTC time
+ position.set(Event.KEY_ODOMETER, buf.readUnsignedInt());
+ buf.readUnsignedInt(); // trip odometer
+ buf.readUnsignedInt(); // total fuel consumption
+ buf.readUnsignedShort(); // current fuel consumption
+ position.set(Event.KEY_STATUS, buf.readUnsignedInt());
+ buf.skipBytes(8);
+
+ buf.readUnsignedByte(); // count
+
+ // Date and time
+ Calendar time = Calendar.getInstance(TimeZone.getTimeZone("UTC"));
+ time.clear();
+ time.set(Calendar.DAY_OF_MONTH, buf.readUnsignedByte());
+ time.set(Calendar.MONTH, buf.readUnsignedByte() - 1);
+ time.set(Calendar.YEAR, 2000 + buf.readUnsignedByte());
+ time.set(Calendar.HOUR_OF_DAY, buf.readUnsignedByte());
+ time.set(Calendar.MINUTE, buf.readUnsignedByte());
+ time.set(Calendar.SECOND, buf.readUnsignedByte());
+ position.setTime(time.getTime());
+
+ double lat = buf.readUnsignedInt() / 3600000.0;
+ double lon = buf.readUnsignedInt() / 3600000.0;
+ position.setSpeed(UnitsConverter.knotsFromKph(buf.readUnsignedShort()));
+ position.setCourse(buf.readUnsignedShort());
+
+ int flags = buf.readUnsignedByte();
+ position.setLatitude((flags & 0x01) == 0 ? -lat : lat);
+ position.setLongitude((flags & 0x02) == 0 ? -lon : lon);
+ position.setValid((flags & 0x0C) > 0);
+ position.set(Event.KEY_SATELLITES, flags >> 4);
+
+ return position;
+ }
+
+ return null;
+ }
+
+}
diff --git a/test/org/traccar/protocol/CastelProtocolDecoderTest.java b/test/org/traccar/protocol/CastelProtocolDecoderTest.java
new file mode 100644
index 000000000..71d34bc5e
--- /dev/null
+++ b/test/org/traccar/protocol/CastelProtocolDecoderTest.java
@@ -0,0 +1,28 @@
+package org.traccar.protocol;
+
+import java.nio.ByteOrder;
+import org.jboss.netty.buffer.ChannelBuffers;
+import static org.junit.Assert.assertNull;
+import org.junit.Test;
+import org.traccar.helper.ChannelBufferTools;
+import static org.traccar.helper.DecoderVerifier.verify;
+
+public class CastelProtocolDecoderTest extends ProtocolDecoderTest {
+
+ @Test
+ public void testDecode() throws Exception {
+
+ CastelProtocolDecoder decoder = new CastelProtocolDecoder(null);
+
+ assertNull(decoder.decode(null, null, null, ChannelBuffers.wrappedBuffer(ByteOrder.LITTLE_ENDIAN, ChannelBufferTools.convertHexString(
+ "40401F00043130303131313235323939383700000000000000100303320D0A"))));
+
+ verify(decoder.decode(null, null, null, ChannelBuffers.wrappedBuffer(ByteOrder.LITTLE_ENDIAN, ChannelBufferTools.convertHexString(
+ "40407F000431303031313132353239393837000000000000001001C1F06952FDF069529C91110000000000698300000C0000000000036401014C00030001190A0D04121A1480D60488C5721800000000AF4944445F3231364730325F532056312E322E31004944445F3231364730325F482056312E322E31000000DF640D0A"))));
+
+ //verify(decoder.decode(null, null, null, ChannelBuffers.wrappedBuffer(ByteOrder.LITTLE_ENDIAN, ChannelBufferTools.convertHexString(
+ // "404057000431303031313132353239393837000000000000004002C1F06952F0F169529C9111000000000069830000470000000400036401014C01030078000505210C210D210F21102101073BE8030064280AEB930D0A"))));
+
+ }
+
+}
diff --git a/test/org/traccar/protocol/MeitrackProtocolDecoderTest.java b/test/org/traccar/protocol/MeitrackProtocolDecoderTest.java
index 339cddde5..6c3aa1b46 100644
--- a/test/org/traccar/protocol/MeitrackProtocolDecoderTest.java
+++ b/test/org/traccar/protocol/MeitrackProtocolDecoderTest.java
@@ -3,7 +3,6 @@ package org.traccar.protocol;
import java.nio.ByteOrder;
import java.nio.charset.Charset;
import org.jboss.netty.buffer.ChannelBuffers;
-import org.traccar.helper.TestDataManager;
import org.junit.Test;
import org.traccar.helper.ChannelBufferTools;
import static org.traccar.helper.DecoderVerifier.verify;
@@ -14,6 +13,9 @@ public class MeitrackProtocolDecoderTest extends ProtocolDecoderTest {
public void testDecode() throws Exception {
MeitrackProtocolDecoder decoder = new MeitrackProtocolDecoder(null);
+
+ verify(decoder.decode(null, null, ChannelBuffers.copiedBuffer(
+ "$$G145,862106024274815,AAA,35,-1.287125,36.906061,150530054639,A,10,13,12,67,0.8,1621,38359791,42330881,639|2|FB2|2F3,0000,3|0|0|A58|432,,,1,0009,*26", Charset.defaultCharset())));
verify(decoder.decode(null, null, ChannelBuffers.copiedBuffer(
"$$I152,013949004569813,AAA,37,54.739468,25.273648,150208173414,A,5,24,0,73,1.5,165,74,3381,246|1|0065|118A,0000,0003|0003|0000|08D4|0002,006380DF,,1,0008*7C", Charset.defaultCharset())));