update vehicle adapter

This commit is contained in:
wait 2025-04-18 11:06:14 +08:00
parent 17c28a5a7f
commit 4882a28c0a
81 changed files with 4660 additions and 4635 deletions

View File

@ -1,40 +0,0 @@
package org.opentcs.kcvehicle;
import com.google.inject.assistedinject.FactoryModuleBuilder;
import org.opentcs.customizations.kernel.KernelInjectionModule;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class KcCommAdapterModule extends KernelInjectionModule {
private static final Logger LOG = LoggerFactory.getLogger(KcCommAdapterModule.class);
@Override
protected void configure() {
//
// VirtualVehicleConfiguration configuration
// = getConfigBindingProvider().get(
// VirtualVehicleConfiguration.PREFIX,
// VirtualVehicleConfiguration.class
// );
//
//// KcVehicleConfiguration configuration
//// = getConfigBindingProvider().get(
//// KcVehicleConfiguration.PREFIX,
//// KcVehicleConfiguration.class
//// );
//
// if (!configuration.enable()) {
// LOG.info("KC driver disabled by configuration.");
// return;
// }
//
// bind(VirtualVehicleConfiguration.class)
// .toInstance(configuration);
//
// install(new FactoryModuleBuilder().build(LoopbackAdapterComponentsFactory.class));
//
// vehicleCommAdaptersBinder().addBinding().to(LoopbackCommunicationAdapterFactory.class);
}
}

View File

@ -2,4 +2,3 @@
# SPDX-License-Identifier: MIT # SPDX-License-Identifier: MIT
org.opentcs.virtualvehicle.LoopbackCommAdapterModule org.opentcs.virtualvehicle.LoopbackCommAdapterModule
org.opentcs.kcvehicle.KcCommAdapterModule

View File

@ -1,10 +0,0 @@
package org.opentcs.kcvehicle;
import org.opentcs.data.model.Vehicle;
import org.opentcs.kcvehicle.KcCommunicationAdapter;
public interface KcAdapterComponentsFactory {
// KcCommunicationAdapter createKcCommunicationAdapter(Vehicle vehicle);
KcCommunicationAdapter createKcCommunicationAdapter(Vehicle vehicle);
}

View File

@ -1,163 +0,0 @@
package org.opentcs.kcvehicle;
import com.google.inject.assistedinject.Assisted;
import jakarta.annotation.Nonnull;
import jakarta.annotation.Nullable;
import jakarta.inject.Inject;
import java.util.concurrent.ScheduledExecutorService;
import org.opentcs.customizations.kernel.KernelExecutor;
import org.opentcs.data.model.Vehicle;
import org.opentcs.data.order.TransportOrder;
import org.opentcs.drivers.vehicle.BasicVehicleCommAdapter;
import org.opentcs.drivers.vehicle.MovementCommand;
import org.opentcs.drivers.vehicle.VehicleProcessModel;
import org.opentcs.kc.udp.Service.ConfirmRelocation;
import org.opentcs.kc.udp.Service.ManualPosition;
import org.opentcs.kc.udp.Service.QryRobotRunStatus;
import org.opentcs.kc.udp.Service.QryRobotStatus;
import org.opentcs.kc.udp.Service.SubCargoStatus;
import org.opentcs.kc.udp.Service.SubRobotStatue;
import org.opentcs.kc.udp.Service.SwitchAutomaticMode;
import org.opentcs.kc.udp.Service.SwitchManualMode;
import org.opentcs.kc.udp.agv.param.function.af.LocationStatusInfo;
import org.opentcs.kc.udp.agv.param.function.af.QueryRobotStatusRsp;
import org.opentcs.kc.udp.agv.param.function.x17.QueryRobotRunStatusRsp;
import org.opentcs.util.ExplainedBoolean;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class KcCommunicationAdapter extends BasicVehicleCommAdapter {
private static final Logger LOG = LoggerFactory.getLogger(KcCommunicationAdapter.class);
/**
* This instance's configuration.
*/
private final KcVehicleConfiguration configuration;
/**
* 单个仿真步骤的时间 毫秒
*/
private static final int SIMULATION_PERIOD = 100;
/**
* Creates a new instance.
*/
@Inject
public KcCommunicationAdapter(
KcVehicleConfiguration configuration,
@Assisted
Vehicle vehicle,
@KernelExecutor
ScheduledExecutorService kernelExecutor
) {
super(
new VehicleProcessModel(vehicle),
configuration.commandQueueCapacity(),
configuration.rechargeOperation(),
kernelExecutor
);
this.configuration = configuration;
}
// @Inject
// public KcCommunicationAdapter(Vehicle vehicle) {
// super(new VehicleProcessModel(vehicle), 1, "Recharge");
// }
@Override
public void sendCommand(MovementCommand cmd)
throws IllegalArgumentException {
}
@Override
protected void connectVehicle() {
//initAGV();
getProcessModel().setCommAdapterConnected(true);
}
@Override
protected void disconnectVehicle() {
getProcessModel().setCommAdapterConnected(false);
}
@Override
protected boolean isVehicleConnected() {
return getProcessModel().isCommAdapterConnected();
}
@Nonnull
@Override
public ExplainedBoolean canProcess(
@Nonnull
TransportOrder order
) {
return null;
}
@Override
public void onVehiclePaused(boolean paused) {
}
@Override
public void processMessage(
@Nullable
Object message
) {
}
/**
* 初始化AGV
* 步骤:
* 1调度软件启动机器人启动无顺序要求
* 2等待调度系统以及机器人控制器启动完成调度系统启动即向机器人发送状态查询查询成功即为启动完成
* 3调度软件发送订阅信令至机器人表明订阅机器人状态信息或载货状态机器人接收到订阅信令会依据订阅要求推送订阅信息度软件需要根据订阅信令中上报持续时间提前刷新机器人推送的上报持续时间
* 4调度软件持续监控机器人实时状态
* 5导航初始化
*/
private void initAGV() {
//0xAF获取AGV状态
QueryRobotStatusRsp qryRobotStatusRsp = QryRobotStatus.command();
//开启AGV订阅监听 (0xAF & 0xB0)
SubRobotStatue.command();
SubCargoStatus.command();
//1 切换定位为手动模式命令码0x03变量NaviControl 修改为0
SwitchManualMode.command();
LocationStatusInfo locationStatusInfo = qryRobotStatusRsp.locationStatusInfo;
double agvX = locationStatusInfo.globalX;
double agvY = locationStatusInfo.globalY;
double agvAngle = locationStatusInfo.absoluteDirecAngle;
//3 查询机器人运行状态命令码0x17等待机器人定位状态为定位完成
QueryRobotRunStatusRsp qryRobotRunStatusRsp = QryRobotRunStatus.command();
if (qryRobotRunStatusRsp.robotLocalizationState == 0) {
//2 执行机器人手动定位 命令码0x14
ManualPosition.command(agvX, agvY, agvAngle);
throw new RuntimeException("AGV定位失败执行手动定位中");
} else if (qryRobotRunStatusRsp.robotLocalizationState == 2) {
throw new RuntimeException("AGV定位中");
}
//4 确认初始位置命令码0x1F
ConfirmRelocation.commnd();
//5 切换成自动模式命令码0x03变量NaviControl 修改为1
SwitchAutomaticMode.command();
//打开通讯适配器连接
getProcessModel().setCommAdapterConnected(true);
}
private int getSimulationTimeStep() {
return (int) (SIMULATION_PERIOD * configuration.simulationTimeFactor());
}
}

View File

@ -1,15 +0,0 @@
package org.opentcs.kcvehicle;
import org.opentcs.drivers.vehicle.VehicleCommAdapterDescription;
public class KcCommunicationAdapterDescription extends VehicleCommAdapterDescription {
@Override
public String getDescription() {
return "KC_ADAPTER";
}
@Override
public boolean isSimVehicleCommAdapter() {
return false;
}
}

View File

@ -1,61 +0,0 @@
package org.opentcs.kcvehicle;
import static java.util.Objects.requireNonNull;
import jakarta.annotation.Nonnull;
import jakarta.annotation.Nullable;
import jakarta.inject.Inject;
import org.opentcs.data.model.Vehicle;
import org.opentcs.drivers.vehicle.VehicleCommAdapter;
import org.opentcs.drivers.vehicle.VehicleCommAdapterDescription;
import org.opentcs.drivers.vehicle.VehicleCommAdapterFactory;
import org.opentcs.drivers.vehicle.VehicleProcessModel;
public class KcCommunicationAdapterFactory implements VehicleCommAdapterFactory {
private final KcAdapterComponentsFactory adapterFactory;
@Inject
public KcCommunicationAdapterFactory(KcAdapterComponentsFactory componentsFactory) {
this.adapterFactory = requireNonNull(componentsFactory, "KC_componentsFactory_NULL");
}
@Override
public VehicleCommAdapterDescription getDescription() {
return new KcCommunicationAdapterDescription();
}
@Override
public boolean providesAdapterFor(
@Nonnull
Vehicle vehicle
) {
return false;
}
@Nullable
@Override
public VehicleCommAdapter getAdapterFor(
@Nonnull
Vehicle vehicle
) {
return adapterFactory.createKcCommunicationAdapter(vehicle);
// return adapterFactory.createKcCommunicationAdapter(vehicle);
}
@Override
public void initialize() {
}
@Override
public boolean isInitialized() {
return false;
}
@Override
public void terminate() {
}
}

View File

@ -1,77 +0,0 @@
// SPDX-FileCopyrightText: The openTCS Authors
// SPDX-License-Identifier: MIT
package org.opentcs.kcvehicle;
import org.opentcs.configuration.ConfigurationEntry;
import org.opentcs.configuration.ConfigurationPrefix;
/**
* Provides methods to configure to {@link KcCommunicationAdapter}.
*/
@ConfigurationPrefix(KcVehicleConfiguration.PREFIX)
public interface KcVehicleConfiguration {
/**
* This configuration's prefix.
*/
String PREFIX = "kcvehicle";
@ConfigurationEntry(
type = "Boolean",
description = "Whether to enable to register/enable the loopback driver.",
changesApplied = ConfigurationEntry.ChangesApplied.ON_APPLICATION_START,
orderKey = "0_enable"
)
boolean enable();
@ConfigurationEntry(
type = "Integer",
description = "The adapter's command queue capacity.",
changesApplied = ConfigurationEntry.ChangesApplied.ON_NEW_PLANT_MODEL,
orderKey = "1_attributes_1"
)
int commandQueueCapacity();
@ConfigurationEntry(
type = "String",
description = "The string to be treated as a recharge operation.",
changesApplied = ConfigurationEntry.ChangesApplied.ON_NEW_PLANT_MODEL,
orderKey = "1_attributes_2"
)
String rechargeOperation();
@ConfigurationEntry(
type = "Double",
description = "The rate at which the vehicle recharges in percent per second.",
changesApplied = ConfigurationEntry.ChangesApplied.INSTANTLY,
orderKey = "1_attributes_3"
)
double rechargePercentagePerSecond();
@ConfigurationEntry(
type = "Double",
description = {
"The simulation time factor.",
"1.0 is real time, greater values speed up simulation."
},
changesApplied = ConfigurationEntry.ChangesApplied.INSTANTLY,
orderKey = "2_behaviour_1"
)
double simulationTimeFactor();
@ConfigurationEntry(
type = "Integer",
description = {"The virtual vehicle's length in mm when it's loaded."},
changesApplied = ConfigurationEntry.ChangesApplied.INSTANTLY,
orderKey = "2_behaviour_2"
)
int vehicleLengthLoaded();
@ConfigurationEntry(
type = "Integer",
description = {"The virtual vehicle's length in mm when it's unloaded."},
changesApplied = ConfigurationEntry.ChangesApplied.INSTANTLY,
orderKey = "2_behaviour_3"
)
int vehicleLengthUnloaded();
}

View File

@ -7,6 +7,7 @@ import static java.util.Objects.requireNonNull;
import com.google.inject.assistedinject.Assisted; import com.google.inject.assistedinject.Assisted;
import jakarta.inject.Inject; import jakarta.inject.Inject;
import java.beans.PropertyChangeEvent; import java.beans.PropertyChangeEvent;
import java.time.LocalDateTime;
import java.util.Arrays; import java.util.Arrays;
import java.util.Date; import java.util.Date;
import java.util.HashMap; import java.util.HashMap;
@ -14,11 +15,14 @@ import java.util.Iterator;
import java.util.List; import java.util.List;
import java.util.Objects; import java.util.Objects;
import java.util.concurrent.ExecutorService; import java.util.concurrent.ExecutorService;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import org.opentcs.common.LoopbackAdapterConstants; import org.opentcs.common.LoopbackAdapterConstants;
import org.opentcs.customizations.kernel.KernelExecutor; import org.opentcs.customizations.kernel.KernelExecutor;
import org.opentcs.data.model.Point;
import org.opentcs.data.model.Pose; import org.opentcs.data.model.Pose;
import org.opentcs.data.model.Triple; import org.opentcs.data.model.Triple;
import org.opentcs.data.model.Vehicle; import org.opentcs.data.model.Vehicle;
@ -44,6 +48,7 @@ import org.opentcs.kc.udp.agv.param.function.af.LocationStatusInfo;
import org.opentcs.kc.udp.agv.param.function.af.QueryRobotStatusRsp; import org.opentcs.kc.udp.agv.param.function.af.QueryRobotStatusRsp;
import org.opentcs.kc.udp.agv.param.function.b0.QueryCargoStatusRsp; import org.opentcs.kc.udp.agv.param.function.b0.QueryCargoStatusRsp;
import org.opentcs.kc.udp.agv.param.function.x17.QueryRobotRunStatusRsp; import org.opentcs.kc.udp.agv.param.function.x17.QueryRobotRunStatusRsp;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.util.ExplainedBoolean; import org.opentcs.util.ExplainedBoolean;
import org.opentcs.virtualvehicle.VelocityController.WayEntry; import org.opentcs.virtualvehicle.VelocityController.WayEntry;
import org.slf4j.Logger; import org.slf4j.Logger;
@ -113,25 +118,32 @@ public class LoopbackCommunicationAdapter
*/ */
private boolean initialized; private boolean initialized;
/** /**
* 上报截止时间 * 0xAF上报截止时间.
*/ */
private long deadline; private long sub0xafDeadline;
/** /**
* 上报间隔时间/ms * 0xB0上报截止时间.
*/ */
private long intervalTime; private long sub0xb0Deadline;
// /** /**
// * AGV IP * 创建线程
// */ */
// private final String IP; private final ExecutorService messageProcessingPool = new ThreadPoolExecutor(
// /** 4,
// * AGV 端口 8,
// */ 60L,
// private final int PORT; TimeUnit.MILLISECONDS,
// /** new LinkedBlockingQueue<>(100),
// * AGV AUTHORIZE_CODE new ThreadPoolExecutor.CallerRunsPolicy()
// */ );
// private final String AUTHORIZE_CODE; /**
* 订阅状态
*/
private static boolean SUBSCRIBE_STATUS;
/**
* 最后经过点位
*/
private static String LAST_PASSED_POINT;
/** /**
* Creates a new instance. * Creates a new instance.
@ -193,41 +205,52 @@ public class LoopbackCommunicationAdapter
@Override @Override
public void propertyChange(PropertyChangeEvent evt) { public void propertyChange(PropertyChangeEvent evt) {
//调用父类的 propertyChange 方法处理事件
super.propertyChange(evt); super.propertyChange(evt);
//如果事件源不是 LoopbackVehicleModel 类型直接返回
if (!((evt.getSource()) instanceof LoopbackVehicleModel)) { if (!((evt.getSource()) instanceof LoopbackVehicleModel)) {
return; return;
} }
//如果事件属性名为 LOAD_HANDLING_DEVICES
if (Objects.equals( if (Objects.equals(
evt.getPropertyName(), evt.getPropertyName(),
VehicleProcessModel.Attribute.LOAD_HANDLING_DEVICES.name() VehicleProcessModel.Attribute.LOAD_HANDLING_DEVICES.name()
)) { )) {
if (!getProcessModel().getLoadHandlingDevices().isEmpty() if (!getProcessModel().getLoadHandlingDevices().isEmpty()
&& getProcessModel().getLoadHandlingDevices().get(0).isFull()) { && getProcessModel().getLoadHandlingDevices().get(0).isFull()) {
//检查负载处理设备是否为空且第一个设备是否满载更新负载状态为 FULL 并设置车辆长度为加载状态下的长度
loadState = LoadState.FULL; loadState = LoadState.FULL;
getProcessModel().setBoundingBox( getProcessModel().setBoundingBox(
getProcessModel().getBoundingBox().withLength(configuration.vehicleLengthLoaded()) getProcessModel().getBoundingBox().withLength(configuration.vehicleLengthLoaded())
); );
} }
else { else {
//否则更新负载状态为 EMPTY 并设置车辆长度为未加载状态下的长度
loadState = LoadState.EMPTY; loadState = LoadState.EMPTY;
getProcessModel().setBoundingBox( getProcessModel().setBoundingBox(
getProcessModel().getBoundingBox().withLength(configuration.vehicleLengthUnloaded()) getProcessModel().getBoundingBox().withLength(configuration.vehicleLengthUnloaded())
); );
} }
} }
//如果事件属性名为 SINGLE_STEP_MODE
if (Objects.equals( if (Objects.equals(
evt.getPropertyName(), evt.getPropertyName(),
LoopbackVehicleModel.Attribute.SINGLE_STEP_MODE.name() LoopbackVehicleModel.Attribute.SINGLE_STEP_MODE.name()
)) { )) {
// When switching from single step mode to automatic mode and there are commands to be // When switching from single step mode to automatic mode and there are commands to be
// processed, ensure that we start/continue processing them. // processed, ensure that we start/continue processing them.
//如果单步模式关闭待处理命令队列非空且模拟未运行则启动车辆模拟
if (!getProcessModel().isSingleStepModeEnabled() if (!getProcessModel().isSingleStepModeEnabled()
&& !getSentCommands().isEmpty() && !getSentCommands().isEmpty()
&& !isSimulationRunning) { && !isSimulationRunning) {
//标记模拟正在运行
isSimulationRunning = true; isSimulationRunning = true;
//提交任务到线程池执行队列中的第一个命令
((ExecutorService) getExecutor()).submit( ((ExecutorService) getExecutor()).submit(
() -> startVehicleSimulation(getSentCommands().peek()) () -> startVehicle(getSentCommands().peek())
); );
} }
} }
@ -261,6 +284,7 @@ public class LoopbackCommunicationAdapter
System.out.println(cmd); System.out.println(cmd);
System.out.println("send cmd print start"); System.out.println("send cmd print start");
// SubRobotStatue.command();
//订单ID //订单ID
String orderName = cmd.getTransportOrder().getName(); String orderName = cmd.getTransportOrder().getName();
@ -283,29 +307,40 @@ public class LoopbackCommunicationAdapter
System.out.println("sourcePoint:" + sourcePoint); System.out.println("sourcePoint:" + sourcePoint);
//下发终点 //下发终点
String destinationPoint = cmd.getStep().getDestinationPoint().getName(); Point destinationPoint = cmd.getStep().getDestinationPoint();
System.out.println("destinationPoint:" + destinationPoint); //获取点类型
String pointProperty = destinationPoint.getProperty(LoopbackAdapterConstants.POINT_TYPE);
System.out.println("destinationPoint_tcs:point:" + pointProperty);
String destinationPointName = cmd.getStep().getDestinationPoint().getName();
System.out.println("destinationPointName:" + destinationPointName);
//拼接起点终点字符串转为Integer类型获取唯一pathID
Integer pathID = Integer.parseInt(sourcePoint + destinationPoint);
System.out.println("pathID:" + pathID);
System.out.println("send cmd print end"); System.out.println("send cmd print end");
//订阅0xAF
// sub0xAF();
//AGV控制器执行命令 //AGV控制器执行命令
HybridNavigation.command("1", sourcePoint, destinationPoint, operation); HybridNavigation.command(orderName, sourcePoint, destinationPointName, operation);
// Start the simulation task if we're not in single step mode and not simulating already. //检查当前车辆模型是否处于单步模式且未运行若满足条件则设置运行状态为true
if (!getProcessModel().isSingleStepModeEnabled() if (!getProcessModel().isSingleStepModeEnabled()
&& !isSimulationRunning) { && !isSimulationRunning) {
isSimulationRunning = true; isSimulationRunning = true;
// The command is added to the sent queue after this method returns. Therefore
// we have to explicitly start the simulation like this. if (LAST_PASSED_POINT == null) {
//设置最后经过点为起点
LAST_PASSED_POINT = sourcePoint;
//设置订阅状态为true
SUBSCRIBE_STATUS = true;
}
sub0xAF();
// 展示模拟车辆
if (getSentCommands().isEmpty()) { if (getSentCommands().isEmpty()) {
((ExecutorService) getExecutor()).submit(() -> startVehicleSimulation(cmd)); ((ExecutorService) getExecutor()).submit(() -> startVehicle(cmd));
} }
else { else {
((ExecutorService) getExecutor()).submit( ((ExecutorService) getExecutor()).submit(
() -> startVehicleSimulation(getSentCommands().peek()) () -> startVehicle(getSentCommands().peek())
); );
} }
} }
@ -318,74 +353,100 @@ public class LoopbackCommunicationAdapter
@Override @Override
public void processMessage(Object message) { public void processMessage(Object message) {
if (message instanceof String) {
//测试使用 if (message instanceof byte[]) {
getProcessModel().setEnergyLevel(66); updateVehicleModel(message);
System.out.println(message);
} }
else if (message instanceof QueryCargoStatusRsp) { else if (message instanceof HashMap<?, ?>) {
//0xB0响应结果 //todo 测试代码----成功
QueryCargoStatusRsp queryCargoStatusRsp = (QueryCargoStatusRsp) message; HashMap<?, ?> msg = (HashMap<?, ?>) message;
if (queryCargoStatusRsp.isCargo == 0) { getProcessModel().setEnergyLevel((int) msg.get("energy"));
this.loadState = LoadState.EMPTY;
} else {
this.loadState = LoadState.FULL;
}
}
else if (message instanceof QueryRobotStatusRsp) {
System.out.println("test success");
//0xAF响应结果
QueryRobotStatusRsp queryRobotStatusRsp = (QueryRobotStatusRsp) message;
//电量
float batteryPercentage = queryRobotStatusRsp.batteryStatusInfo.batteryPercentage;
getProcessModel().setEnergyLevel(((int)batteryPercentage * 100));
//充电状态
byte chargingState = queryRobotStatusRsp.batteryStatusInfo.chargingState;
if (chargingState == 1) {
getProcessModel().setState(Vehicle.State.CHARGING);
}
//设置车辆位置(最后一次通过的点)
// String lastPointName = queryRobotStatusRsp.locationStatusInfo.lastPassPointId.toString();
// getProcessModel().setPosition(lastPointName);
////设置车辆姿势(官方弃用设置车辆精确位置)
long positionX = (long)queryRobotStatusRsp.locationStatusInfo.globalX;
long positionY = (long)queryRobotStatusRsp.locationStatusInfo.globalY;
Triple triple = new Triple(positionX, positionY, 0);
double positionAngle = queryRobotStatusRsp.locationStatusInfo.absoluteDirecAngle;
getProcessModel().setPose(new Pose(triple, positionAngle));
}
else if (message instanceof HashMap<?,?>) {
HashMap<?,?> msg = ((HashMap<?, ?>) message);
getProcessModel().setEnergyLevel((int)msg.get("energy"));
getProcessModel().setState(Vehicle.State.EXECUTING); getProcessModel().setState(Vehicle.State.EXECUTING);
long positionX = (long)msg.get("positionX"); long positionX = (long) msg.get("positionX");
long positionY = (long)msg.get("positionY"); long positionY = (long) msg.get("positionY");
Triple triple = new Triple(positionX, positionY, 0); Triple triple = new Triple(positionX, positionY, 0);
double positionAngle = (double)msg.get("positionAngle"); double positionAngle = (double) msg.get("positionAngle");
getProcessModel().setPose(new Pose(triple, positionAngle)); getProcessModel().setPose(new Pose(triple, positionAngle));
} }
}
//上报自动续订操作 private void updateVehicleModel(Object message) {
// renewalSubscribe(); try {
byte[] body = (byte[])message;
RcvEventPackage rcv = new RcvEventPackage(body[22], body);
if (body[21] == (byte) 0xAF) {
System.out.println("0xAF sub success");
//AGV状态订阅
QueryRobotStatusRsp queryRobotStatusRsp = new QueryRobotStatusRsp(rcv.getDataBytes());
System.out.println();
//电量--目前无电量返回设置一个随机值
float batteryPercentage = queryRobotStatusRsp.batteryStatusInfo.batteryPercentage;
getProcessModel().setEnergyLevel(89);
//设置AGV最后一个点位置,不设置最后经过点opentcs无法调度
String vehicleNowPosition = getProcessModel().getPosition();
String lastPassPointId = (queryRobotStatusRsp.locationStatusInfo.lastPassPointId).toString();
if (vehicleNowPosition == null || !vehicleNowPosition.equals(lastPassPointId)) {
initVehiclePosition(lastPassPointId);
}
//设置车辆姿势(官方弃用设置车辆精确位置)-------------------目前车辆返回位置为固定值
/* long positionX = (long) queryRobotStatusRsp.locationStatusInfo.globalX;
long positionY = (long) queryRobotStatusRsp.locationStatusInfo.globalY;
Triple triple = new Triple(positionX, positionY, 0);
double positionAngle = queryRobotStatusRsp.locationStatusInfo.absoluteDirecAngle;
LocalDateTime now = LocalDateTime.now();
System.out.println(now + "[positionX:" + positionX + "] [positionY:" + positionY + "] [positionAngle:" + positionAngle + "]");
getProcessModel().setPose(new Pose(triple, positionAngle));*/
//到期续订
renewalSubscribe0xAF();
} else if (body[21] == (byte) 0xB0) {
System.out.println("0xB0 sub success");
//载货状态订阅
QueryCargoStatusRsp queryCargoStatusRsp = new QueryCargoStatusRsp(rcv.getDataBytes());
if (queryCargoStatusRsp.isCargo == 0) {
this.loadState = LoadState.EMPTY;
}
else {
this.loadState = LoadState.FULL;
}
//到期续订
renewalSubscribe0xB0();
}
} catch (Exception e) {
throw new RuntimeException("processMessage_messageExecutorPool:" + e);
}
} }
/** /**
* 订阅到期自动续订 * 订阅到期自动续订0xAF.
*/ */
private void renewalSubscribe() { private void renewalSubscribe0xAF() {
Date now = new Date(); Date now = new Date();
if (now.getTime() + intervalTime >= deadline) { if (sub0xafDeadline - now.getTime() <= SubRobotStatue.intervalTime && SUBSCRIBE_STATUS) {
//开启AGV订阅监听 (0xAF & 0xB0) sub0xAF();
SubRobotStatue.command(); }
SubCargoStatus.command(); }
/**
* 订阅到期自动续订0xB0.
*/
private void renewalSubscribe0xB0() {
Date now = new Date();
if (sub0xb0Deadline - now.getTime() <= SubCargoStatus.intervalTime && SUBSCRIBE_STATUS) {
sub0xB0();
} }
} }
@ -446,11 +507,8 @@ public class LoopbackCommunicationAdapter
@Override @Override
protected synchronized void connectVehicle() { protected synchronized void connectVehicle() {
String ip = requireNonNull(vehicle.getProperties().get(LoopbackAdapterConstants.AGV_IP), "AGV IP NOT NULL"); // getProcessModel().setCommAdapterConnected(true);
Integer port = Integer.parseInt(requireNonNull(vehicle.getProperties().get(LoopbackAdapterConstants.AGV_PORT), "AGV PORT NOT NULL")); initAGV();
String authorizeCode = requireNonNull(vehicle.getProperties().get(LoopbackAdapterConstants.AGV_AUTHORIZE_CODE), "AGV AUTHORIZE_CODE NOT NULL");
getProcessModel().setCommAdapterConnected(true);
// initAGV();
} }
@Override @Override
@ -491,6 +549,36 @@ public class LoopbackCommunicationAdapter
} }
} }
/**
* 执行车辆移动指令
*
* @param command 移动指令
*/
private void startVehicle(MovementCommand command) {
LOG.debug("-Starting vehicle for command: {}", command);
Step step = command.getStep();
getProcessModel().setState(Vehicle.State.EXECUTING);
if (step.getPath() == null) {
LOG.debug("-Starting operation...");
//动作执行待完成
operationExec(command);
} else {
getProcessModel().getVelocityController().addWayEntry(
new WayEntry(
step.getPath().getLength(),
maxVelocity(step),
step.getDestinationPoint().getName(),
step.getVehicleOrientation()
)
);
LOG.debug("-Starting movement ...");
movementExec(command);
}
}
private void startVehicleSimulation(MovementCommand command) { private void startVehicleSimulation(MovementCommand command) {
LOG.debug("Starting vehicle simulation for command: {}", command); LOG.debug("Starting vehicle simulation for command: {}", command);
Step step = command.getStep(); Step step = command.getStep();
@ -529,12 +617,54 @@ public class LoopbackCommunicationAdapter
: step.getPath().getMaxVelocity(); : step.getPath().getMaxVelocity();
} }
/**
* 执行运行指令.
*
* @param command 要执行的命令
*/
private void movementExec(MovementCommand command) {
//检查当前车辆模型的速度控制器是否有路径条目若无则直接返回
if (!getProcessModel().getVelocityController().hasWayEntries()) {
return;
}
//获取AGV最终经过点
String currentPoint = getProcessModel().getPosition() != null ? getProcessModel().getPosition() : "";
//获取当前路径条目并推进时间步长检查是否仍处于同一路径条目
if (currentPoint.equals(LAST_PASSED_POINT)) {
//若是则重新调度当前方法以继续模拟
getExecutor().schedule(
() -> movementExec(command),
SIMULATION_PERIOD,
TimeUnit.MILLISECONDS
);
} else {
LAST_PASSED_POINT = currentPoint;
//若否更新车辆位置为上一路径条目的目标点并根据命令是否有操作决定进入操作模拟或完成命令并模拟下一个命令
LOG.debug("-Movement finished.");
if (!command.hasEmptyOperation()) {
//执行AGV动作
LOG.debug("-Starting operation...");
operationExec(command);
} else {
//完成当前命令
finishMoveCmd(command);
//执行下一个命令
nextCommand();
}
}
}
/** /**
* Simulate the movement part of a MovementCommand. * Simulate the movement part of a MovementCommand.
* *
* @param command The command to simulate. * @param command The command to simulate.
*/ */
private void movementSimulation(MovementCommand command) { private void movementSimulation(MovementCommand command) {
//检查当前车辆模型的速度控制器是否有路径条目若无则直接返回
if (!getProcessModel().getVelocityController().hasWayEntries()) { if (!getProcessModel().getVelocityController().hasWayEntries()) {
return; return;
} }
@ -543,7 +673,9 @@ public class LoopbackCommunicationAdapter
getProcessModel().getVelocityController().advanceTime(getSimulationTimeStep()); getProcessModel().getVelocityController().advanceTime(getSimulationTimeStep());
WayEntry currentWayEntry = getProcessModel().getVelocityController().getCurrentWayEntry(); WayEntry currentWayEntry = getProcessModel().getVelocityController().getCurrentWayEntry();
//if we are still on the same way entry then reschedule to do it again //if we are still on the same way entry then reschedule to do it again
//获取当前路径条目并推进时间步长检查是否仍处于同一路径条目
if (prevWayEntry == currentWayEntry) { if (prevWayEntry == currentWayEntry) {
//若是则重新调度当前方法以继续模拟
getExecutor().schedule( getExecutor().schedule(
() -> movementSimulation(command), () -> movementSimulation(command),
SIMULATION_PERIOD, SIMULATION_PERIOD,
@ -553,6 +685,7 @@ public class LoopbackCommunicationAdapter
else { else {
//if the way enties are different then we have finished this step //if the way enties are different then we have finished this step
//and we can move on. //and we can move on.
//若否更新车辆位置为上一路径条目的目标点并根据命令是否有操作决定进入操作模拟或完成命令并模拟下一个命令
getProcessModel().setPosition(prevWayEntry.getDestPointName()); getProcessModel().setPosition(prevWayEntry.getDestPointName());
LOG.debug("Movement simulation finished."); LOG.debug("Movement simulation finished.");
if (!command.hasEmptyOperation()) { if (!command.hasEmptyOperation()) {
@ -570,8 +703,18 @@ public class LoopbackCommunicationAdapter
} }
} }
/**
* 执行移动命令的操作部分
*
* @param command 要执行的命令
*/
private void operationExec(MovementCommand command) {
}
/** /**
* Simulate the operation part of a movement command. * Simulate the operation part of a movement command.
* 模拟移动命令的操作部分
* *
* @param command The command to simulate. * @param command The command to simulate.
* @param timePassed The amount of time passed since starting the simulation. * @param timePassed The amount of time passed since starting the simulation.
@ -626,6 +769,7 @@ public class LoopbackCommunicationAdapter
/** /**
* Simulate recharging the vehicle. * Simulate recharging the vehicle.
* 模拟为车辆充电
* *
* @param rechargePosition The vehicle position where the recharge simulation was started. * @param rechargePosition The vehicle position where the recharge simulation was started.
* @param rechargePercentage The recharge percentage of the vehicle while it is charging. * @param rechargePercentage The recharge percentage of the vehicle while it is charging.
@ -671,14 +815,43 @@ public class LoopbackCommunicationAdapter
+ (float) (configuration.rechargePercentagePerSecond() / 1000.0) * SIMULATION_PERIOD; + (float) (configuration.rechargePercentagePerSecond() / 1000.0) * SIMULATION_PERIOD;
} }
/**
* 结束移动命令
* @param command 指令
*/
private void finishMoveCmd(MovementCommand command) {
//检查已发送命令队列的大小是否小于等于1且未发送命令队列是否为空
if (getSentCommands().size() <= 1 && getUnsentCommands().isEmpty()) {
System.out.println("-getSentCommands <= 1 && getUnsentCommands is null");
getProcessModel().setState(Vehicle.State.IDLE);
//清除订单对应唯一ID
HybridNavigation.delUniqueOrderID(command);
}
//如果传入指令和移动指令队列第一条数据相同
if (Objects.equals(getSentCommands().peek(), command)) {
// 完成当前任务
getProcessModel().commandExecuted(getSentCommands().poll());
} else {
//
LOG.warn(
"-{}: Exec command not oldest in sent queue: {} != {}",
getName(),
command,
getSentCommands().peek()
);
}
}
private void finishMovementCommand(MovementCommand command) { private void finishMovementCommand(MovementCommand command) {
//Set the vehicle state to idle //Set the vehicle state to idle
if (getSentCommands().size() <= 1 && getUnsentCommands().isEmpty()) { if (getSentCommands().size() <= 1 && getUnsentCommands().isEmpty()) {
getProcessModel().setState(Vehicle.State.IDLE); getProcessModel().setState(Vehicle.State.IDLE);
} }
if (Objects.equals(getSentCommands().peek(), command)) { if (Objects.equals(getSentCommands().peek(), command)) {
// Let the comm adapter know we have finished this command. // Let the comm adapter know we have finished this command. 让通信适配器知道我们已经完成了这个命令
getProcessModel().commandExecuted(getSentCommands().poll()); getProcessModel().commandExecuted(getSentCommands().poll());
// HybridNavigation.delUniqueOrderID(command);
} }
else { else {
LOG.warn( LOG.warn(
@ -690,6 +863,22 @@ public class LoopbackCommunicationAdapter
} }
} }
void nextCommand() {
if (getSentCommands().isEmpty() || getProcessModel().isSingleStepModeEnabled()) {
LOG.debug("Vehicle exec is done.");
getProcessModel().setState(Vehicle.State.IDLE);
isSimulationRunning = false;
LAST_PASSED_POINT = null;
SUBSCRIBE_STATUS = false;
}
else {
LOG.debug("Triggering exec for next command: {}", getSentCommands().peek());
((ExecutorService) getExecutor()).submit(
() -> startVehicle(getSentCommands().peek())
);
}
}
void simulateNextCommand() { void simulateNextCommand() {
if (getSentCommands().isEmpty() || getProcessModel().isSingleStepModeEnabled()) { if (getSentCommands().isEmpty() || getProcessModel().isSingleStepModeEnabled()) {
LOG.debug("Vehicle simulation is done."); LOG.debug("Vehicle simulation is done.");
@ -727,47 +916,133 @@ public class LoopbackCommunicationAdapter
*/ */
private void initAGV() { private void initAGV() {
if (true) {
//0xAF获取AGV状态 //0xAF获取AGV状态
System.out.println("=================---initAGV_0xAF");
QueryRobotStatusRsp qryRobotStatusRsp = QryRobotStatus.command(); QueryRobotStatusRsp qryRobotStatusRsp = QryRobotStatus.command();
if (qryRobotStatusRsp == null) {
getProcessModel().setCommAdapterConnected(false);
throw new RuntimeException("initAGV 0xAF response is null");
}
//设置订阅时间和上报间隔 //开启0xAF订阅
Date now = new Date(); sub0xAF();
deadline = now.getTime() + 1000;
intervalTime = 100;
//开启AGV订阅监听 (0xAF & 0xB0)
SubRobotStatue.command(); //开启0xB0订阅
SubCargoStatus.command(); // sub0xB0()
// //设置车辆姿势(官方弃用设置车辆精确位置)-------------------目前车辆返回位置为固定值
// long positionX = (long) qryRobotStatusRsp.locationStatusInfo.globalX;
// long positionY = (long) qryRobotStatusRsp.locationStatusInfo.globalY;
// Triple triple = new Triple(positionX, positionY, 0);
// double positionAngle = qryRobotStatusRsp.locationStatusInfo.absoluteDirecAngle;
// getProcessModel().setPose(new Pose(triple, positionAngle));
//3 查询机器人运行状态命令码0x17等待机器人定位状态为定位完成
System.out.println("=================---initAGV_0x17");
QueryRobotRunStatusRsp qryRobotRunStatusRsp = QryRobotRunStatus.command();
if (qryRobotRunStatusRsp == null) {
getProcessModel().setCommAdapterConnected(false);
throw new RuntimeException("initAGV 0x17 response is null");
}
if (qryRobotRunStatusRsp.robotLocalizationState == 0) {
getProcessModel().setCommAdapterConnected(false);
throw new RuntimeException("AGV定位失败请执行手动定位");
}
else if (qryRobotRunStatusRsp.robotLocalizationState == 2) {
getProcessModel().setCommAdapterConnected(false);
throw new RuntimeException("AGV定位中,请稍后再试");
}
//5 切换成自动模式命令码0x03变量NaviControl 修改为1
System.out.println();
System.out.println("=================---initAGV_0x03-----111111");
SwitchAutomaticMode.command();
getProcessModel().setState(Vehicle.State.IDLE);
//打开通讯适配器连接
getProcessModel().setCommAdapterConnected(true);
} else {
//0xAF获取AGV状态
System.out.println("=================---initAGV_0xAF");
QueryRobotStatusRsp qryRobotStatusRsp = QryRobotStatus.command();
if (qryRobotStatusRsp == null) {
getProcessModel().setCommAdapterConnected(false);
throw new RuntimeException("initAGV 0xAF response is null");
}
//开启0xAF订阅
sub0xAF();
//开启0xB0订阅
sub0xB0();
//1 切换定位为手动模式命令码0x03变量NaviControl 修改为0 //1 切换定位为手动模式命令码0x03变量NaviControl 修改为0
System.out.println();
System.out.println("=================---initAGV_0x03-----000000");
SwitchManualMode.command(); SwitchManualMode.command();
getProcessModel().setState(Vehicle.State.UNAVAILABLE);
//设置AGV最后一个点位置
String lastPassPointId = (qryRobotStatusRsp.locationStatusInfo.lastPassPointId).toString();
initVehiclePosition(lastPassPointId);
//2 执行机器人手动定位 命令码0x14
System.out.println("=================---initAGV_0x14");
LocationStatusInfo locationStatusInfo = qryRobotStatusRsp.locationStatusInfo; LocationStatusInfo locationStatusInfo = qryRobotStatusRsp.locationStatusInfo;
double agvX = locationStatusInfo.globalX; double agvX = locationStatusInfo.globalX;
double agvY = locationStatusInfo.globalY; double agvY = locationStatusInfo.globalY;
double agvAngle = locationStatusInfo.absoluteDirecAngle; double agvAngle = locationStatusInfo.absoluteDirecAngle;
//3 查询机器人运行状态命令码0x17等待机器人定位状态为定位完成
QueryRobotRunStatusRsp qryRobotRunStatusRsp = QryRobotRunStatus.command();
if (qryRobotRunStatusRsp.robotLocalizationState == 0) {
//2 执行机器人手动定位 命令码0x14
ManualPosition.command(agvX, agvY, agvAngle); ManualPosition.command(agvX, agvY, agvAngle);
throw new RuntimeException("AGV定位失败执行手动定位中"); //3 查询机器人运行状态命令码0x17等待机器人定位状态为定位完成
System.out.println("=================---initAGV_0x17");
QueryRobotRunStatusRsp qryRobotRunStatusRsp = QryRobotRunStatus.command();
} else if (qryRobotRunStatusRsp.robotLocalizationState == 2) { if (qryRobotRunStatusRsp == null) {
throw new RuntimeException("AGV定位中"); getProcessModel().setCommAdapterConnected(false);
throw new RuntimeException("initAGV 0x17 response is null");
}
if (qryRobotRunStatusRsp.robotLocalizationState == 0) {
getProcessModel().setCommAdapterConnected(false);
throw new RuntimeException("AGV定位失败执行手动定位中");
}
else if (qryRobotRunStatusRsp.robotLocalizationState == 2) {
getProcessModel().setCommAdapterConnected(false);
throw new RuntimeException("AGV定位中,请稍后再试");
} }
//4 确认初始位置命令码0x1F //4 确认初始位置命令码0x1F
System.out.println("=================---initAGV_0x1F");
ConfirmRelocation.commnd(); ConfirmRelocation.commnd();
//5 切换成自动模式命令码0x03变量NaviControl 修改为1 //5 切换成自动模式命令码0x03变量NaviControl 修改为1
System.out.println("=================---initAGV_0x03-----111111");
SwitchAutomaticMode.command(); SwitchAutomaticMode.command();
getProcessModel().setState(Vehicle.State.IDLE);
//打开通讯适配器连接 //打开通讯适配器连接
getProcessModel().setCommAdapterConnected(true); getProcessModel().setCommAdapterConnected(true);
} }
}
private void sub0xAF(){
messageProcessingPool.submit(() -> {
Date now = new Date();
sub0xafDeadline = now.getTime() + 10000;
SubRobotStatue.command();
});
}
private void sub0xB0(){
messageProcessingPool.submit(() -> {
Date now = new Date();
sub0xb0Deadline = now.getTime() + 10000;
SubCargoStatus.command();
});
}
} }

View File

@ -58,6 +58,11 @@ public class LoopbackVehicleModelTO
return singleStepModeEnabled; return singleStepModeEnabled;
} }
/**
* 设置单步模式已启用.
* @param singleStepModeEnabled 启用单步模式
* @return LoopbackVehicleModelTO
*/
public LoopbackVehicleModelTO setSingleStepModeEnabled(boolean singleStepModeEnabled) { public LoopbackVehicleModelTO setSingleStepModeEnabled(boolean singleStepModeEnabled) {
this.singleStepModeEnabled = singleStepModeEnabled; this.singleStepModeEnabled = singleStepModeEnabled;
return this; return this;
@ -67,6 +72,11 @@ public class LoopbackVehicleModelTO
return loadOperation; return loadOperation;
} }
/**
* 设置装载操作.
* @param loadOperation 状态
* @return LoopbackVehicleModelTO
*/
public LoopbackVehicleModelTO setLoadOperation(String loadOperation) { public LoopbackVehicleModelTO setLoadOperation(String loadOperation) {
this.loadOperation = loadOperation; this.loadOperation = loadOperation;
return this; return this;
@ -76,6 +86,11 @@ public class LoopbackVehicleModelTO
return unloadOperation; return unloadOperation;
} }
/**
* 设置卸载操作.
* @param unloadOperation 状态
* @return LoopbackVehicleModelTO
*/
public LoopbackVehicleModelTO setUnloadOperation(String unloadOperation) { public LoopbackVehicleModelTO setUnloadOperation(String unloadOperation) {
this.unloadOperation = unloadOperation; this.unloadOperation = unloadOperation;
return this; return this;
@ -85,6 +100,11 @@ public class LoopbackVehicleModelTO
return operatingTime; return operatingTime;
} }
/**
* 设置运行时间.
* @param operatingTime 时间戳
* @return LoopbackVehicleModelTO
*/
public LoopbackVehicleModelTO setOperatingTime(int operatingTime) { public LoopbackVehicleModelTO setOperatingTime(int operatingTime) {
this.operatingTime = operatingTime; this.operatingTime = operatingTime;
return this; return this;
@ -94,6 +114,11 @@ public class LoopbackVehicleModelTO
return maxAcceleration; return maxAcceleration;
} }
/**
* 设置最大加速度.
* @param maxAcceleration
* @return LoopbackVehicleModelTO
*/
public LoopbackVehicleModelTO setMaxAcceleration(int maxAcceleration) { public LoopbackVehicleModelTO setMaxAcceleration(int maxAcceleration) {
this.maxAcceleration = maxAcceleration; this.maxAcceleration = maxAcceleration;
return this; return this;
@ -103,6 +128,11 @@ public class LoopbackVehicleModelTO
return maxDeceleration; return maxDeceleration;
} }
/**
* 设置最大减速.
* @param maxDeceleration
* @return LoopbackVehicleModelTO
*/
public LoopbackVehicleModelTO setMaxDeceleration(int maxDeceleration) { public LoopbackVehicleModelTO setMaxDeceleration(int maxDeceleration) {
this.maxDeceleration = maxDeceleration; this.maxDeceleration = maxDeceleration;
return this; return this;
@ -112,6 +142,11 @@ public class LoopbackVehicleModelTO
return maxFwdVelocity; return maxFwdVelocity;
} }
/**
* 设置最大前驱速度.
* @param maxFwdVelocity
* @return LoopbackVehicleModelTO
*/
public LoopbackVehicleModelTO setMaxFwdVelocity(int maxFwdVelocity) { public LoopbackVehicleModelTO setMaxFwdVelocity(int maxFwdVelocity) {
this.maxFwdVelocity = maxFwdVelocity; this.maxFwdVelocity = maxFwdVelocity;
return this; return this;
@ -121,6 +156,11 @@ public class LoopbackVehicleModelTO
return maxRevVelocity; return maxRevVelocity;
} }
/**
* 设置最大速度.
* @param maxRevVelocity
* @return LoopbackVehicleModelTO
*/
public LoopbackVehicleModelTO setMaxRevVelocity(int maxRevVelocity) { public LoopbackVehicleModelTO setMaxRevVelocity(int maxRevVelocity) {
this.maxRevVelocity = maxRevVelocity; this.maxRevVelocity = maxRevVelocity;
return this; return this;
@ -130,6 +170,11 @@ public class LoopbackVehicleModelTO
return vehiclePaused; return vehiclePaused;
} }
/**
* 将车辆设置为暂停.
* @param vehiclePaused
* @return LoopbackVehicleModelTO
*/
public LoopbackVehicleModelTO setVehiclePaused(boolean vehiclePaused) { public LoopbackVehicleModelTO setVehiclePaused(boolean vehiclePaused) {
this.vehiclePaused = vehiclePaused; this.vehiclePaused = vehiclePaused;
return this; return this;

View File

@ -18,7 +18,7 @@ public interface GuestUserCredentials {
/** /**
* 主机IP * 主机IP
*/ */
String IP = "192.168.0.106"; String IP = "192.168.124.111";
/** /**
* 内核开放端口 * 内核开放端口
*/ */

View File

@ -53,5 +53,9 @@ public interface LoopbackAdapterConstants {
* AGV 端口 * AGV 端口
*/ */
String AGV_PORT = "AGV:PORT"; String AGV_PORT = "AGV:PORT";
/**
* 定制点位类型1=WMS请求点位
*/
String POINT_TYPE = "tcs:point";
} }

View File

@ -56,18 +56,19 @@ public class CaffeineUtil {
} }
public static synchronized byte[] getUUID() { public static synchronized byte[] getUUID() {
AtomicInteger uuid = cacheUUID.getIfPresent("UUID"); AtomicInteger uuid = cacheUUID.getIfPresent("UUID");
if(uuid == null){ if (uuid == null) {
//transationId 从1 开始0留给心跳变量这样就固定报文了 //transationId 从1 开始0留给心跳变量这样就固定报文了
cacheUUID.put("UUID",new AtomicInteger(1)); cacheUUID.put("UUID", new AtomicInteger(1));
return ByteUtils.intToBytes(1); return ByteUtils.intToBytes(1);
}else { }
if(uuid.get() >= 32000){ else {
cacheUUID.put("UUID",new AtomicInteger(1)); if (uuid.get() >= 32000) {
cacheUUID.put("UUID", new AtomicInteger(1));
return ByteUtils.intToBytes(1); return ByteUtils.intToBytes(1);
}else { }
else {
return ByteUtils.intToBytes(uuid.incrementAndGet()); return ByteUtils.intToBytes(uuid.incrementAndGet());
} }
} }
@ -75,15 +76,17 @@ public class CaffeineUtil {
public static synchronized byte[] getUUIDAGV() { public static synchronized byte[] getUUIDAGV() {
AtomicInteger agvuuid = cacheUUID.getIfPresent("AGVUUID"); AtomicInteger agvuuid = cacheUUID.getIfPresent("AGVUUID");
if(agvuuid == null){ if (agvuuid == null) {
//transationId 从1 开始0留给心跳变量这样就固定报文了 //transationId 从1 开始0留给心跳变量这样就固定报文了
cacheUUID.put("AGVUUID",new AtomicInteger(1)); cacheUUID.put("AGVUUID", new AtomicInteger(1));
return ByteUtils.intToBytesS(1); return ByteUtils.intToBytesS(1);
}else { }
if(agvuuid.get() >= 32000){ else {
cacheUUID.put("AGVUUID",new AtomicInteger(1)); if (agvuuid.get() >= 32000) {
cacheUUID.put("AGVUUID", new AtomicInteger(1));
return ByteUtils.intToBytesS(1); return ByteUtils.intToBytesS(1);
}else { }
else {
return ByteUtils.intToBytesS(agvuuid.incrementAndGet()); return ByteUtils.intToBytesS(agvuuid.incrementAndGet());
} }
} }

View File

@ -12,7 +12,8 @@ public interface Const {
public static final String SERVER = "127.0.0.1"; public static final String SERVER = "127.0.0.1";
public static void main(String[] args) { public static void main(String[] args) {
String s = "123, 34, 104, 101, 97, 100, 101, 114, 34, 58, 123, 34, 116, 114, 97, 110, 115, 97, 99, 116, 105, 111, 110, 73, 100, 34, 58, 34, 50, 48, 50, 52, 48, 49, 48, 50, 49, 53, 52, 48, 50, 55, 95, 51, 100, 56, 49, 99, 34, 44, 34, 109, 101, 115, 115, 97, 103, 101, 84, 121, 112, 101, 34, 58, 51, 44, 34, 109, 101, 115, 115, 97, 103, 101, 78, 97, 109, 101, 34, 58, 34, 72, 101, 97, 114, 116, 66, 101, 97, 116, 34, 44, 34, 115, 101, 110, 100, 84, 105, 109, 101, 115, 116, 97, 109, 112, 34, 58, 34, 50, 48, 50, 52, 45, 48, 49, 45, 48, 50, 32, 49, 53, 58, 52, 48, 58, 50, 55, 34, 125, 44, 34, 98, 111, 100, 121, 34, 58, 34, 49, 34, 44, 34, 114, 101, 116, 117, 114, 110, 115, 34, 58, 110, 117, 108, 108, 125"; String s
= "123, 34, 104, 101, 97, 100, 101, 114, 34, 58, 123, 34, 116, 114, 97, 110, 115, 97, 99, 116, 105, 111, 110, 73, 100, 34, 58, 34, 50, 48, 50, 52, 48, 49, 48, 50, 49, 53, 52, 48, 50, 55, 95, 51, 100, 56, 49, 99, 34, 44, 34, 109, 101, 115, 115, 97, 103, 101, 84, 121, 112, 101, 34, 58, 51, 44, 34, 109, 101, 115, 115, 97, 103, 101, 78, 97, 109, 101, 34, 58, 34, 72, 101, 97, 114, 116, 66, 101, 97, 116, 34, 44, 34, 115, 101, 110, 100, 84, 105, 109, 101, 115, 116, 97, 109, 112, 34, 58, 34, 50, 48, 50, 52, 45, 48, 49, 45, 48, 50, 32, 49, 53, 58, 52, 48, 58, 50, 55, 34, 125, 44, 34, 98, 111, 100, 121, 34, 58, 34, 49, 34, 44, 34, 114, 101, 116, 117, 114, 110, 115, 34, 58, 110, 117, 108, 108, 125";
String[] split = s.split(","); String[] split = s.split(",");
System.out.println(split.length); System.out.println(split.length);
} }

View File

@ -3,11 +3,6 @@ package org.opentcs.kc.common;
import java.util.Arrays; import java.util.Arrays;
/**
* @Desc: "通用的数据传输底层类"
* @Author: caixiang
* @DATE: 2022/10/18 16:22
*/
public class Package { public class Package {
private byte[] body; private byte[] body;

View File

@ -1,24 +1,26 @@
/* /*
Copyright 2016 S7connector members (github.com/s7connector) * Copyright 2016 S7connector members (github.com/s7connector)
*
Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
You may obtain a copy of the License at * You may obtain a copy of the License at
*
http://www.apache.org/licenses/LICENSE-2.0 * http://www.apache.org/licenses/LICENSE-2.0
*
Unless required by applicable law or agreed to in writing, software * Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS, * distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and * See the License for the specific language governing permissions and
limitations under the License. * limitations under the License.
*/ */
package org.opentcs.kc.common; package org.opentcs.kc.common;
/** /**
* The Class S7Exception is an exception related to S7 Communication * The Class S7Exception is an exception related to S7 Communication
*/ */
public final class ParseDataException extends RuntimeException { public final class ParseDataException
extends
RuntimeException {
/** The Constant serialVersionUID. */ /** The Constant serialVersionUID. */
private static final long serialVersionUID = -4761415733559374116L; private static final long serialVersionUID = -4761415733559374116L;

View File

@ -1,11 +1,6 @@
package org.opentcs.kc.common; package org.opentcs.kc.common;
/**
* @Desc: "通用的数据传输底层类"
* @Author: caixiang
* @DATE: 2022/10/18 16:22
*/
public class RcvPackage { public class RcvPackage {
private boolean isOk; private boolean isOk;
@ -17,6 +12,7 @@ public class RcvPackage {
this.value = value; this.value = value;
this.content = content; this.content = content;
} }
public RcvPackage() { public RcvPackage() {
} }
@ -37,7 +33,6 @@ public class RcvPackage {
} }
@Override @Override
public String toString() { public String toString() {
return "RcvPackage{" + return "RcvPackage{" +

View File

@ -6,16 +6,8 @@ import java.time.format.DateTimeFormatter;
import java.util.Random; import java.util.Random;
import java.util.UUID; import java.util.UUID;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2022/10/24 16:27
*/
public class Utils { public class Utils {
/**
* java生成随机数字15位数
* @return
*/
public static String getTransationId() { public static String getTransationId() {
String val = String.valueOf(System.currentTimeMillis()); String val = String.valueOf(System.currentTimeMillis());
Random random = new Random(); Random random = new Random();
@ -31,18 +23,15 @@ public class Utils {
* *
* @return * @return
*/ */
public static String getUUID(int len) public static String getUUID(int len) {
{ if (0 >= len) {
if(0 >= len)
{
return null; return null;
} }
String uuid = getUUID(); String uuid = getUUID();
StringBuffer str = new StringBuffer(); StringBuffer str = new StringBuffer();
for (int i = 0; i < len; i++) for (int i = 0; i < len; i++) {
{
str.append(uuid.charAt(i)); str.append(uuid.charAt(i));
} }
@ -50,28 +39,26 @@ public class Utils {
} }
/** /**
*32位默认长度的uuid * 32位默认长度的uuid
* (获取32位uuid) * (获取32位uuid)
* *
* @return * @return
*/ */
public static String getUUID() public static String getUUID() {
{
return UUID.randomUUID().toString().replace("-", ""); return UUID.randomUUID().toString().replace("-", "");
} }
/* /*
type: * type:
1.返回的是这种格式2021-08-16 15:00:05 * 1.返回的是这种格式2021-08-16 15:00:05
2.返回的是这种格式20210816150021 * 2.返回的是这种格式20210816150021
*/ */
public static String getNowDate(Integer type){ public static String getNowDate(Integer type) {
if(type==1){ if (type == 1) {
return LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")); return LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
}else { }
else {
return LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyyMMddHHmmss")); return LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyyMMddHHmmss"));
} }
} }

View File

@ -4,12 +4,7 @@ import java.nio.ByteBuffer;
import java.nio.ByteOrder; import java.nio.ByteOrder;
import java.nio.charset.StandardCharsets; import java.nio.charset.StandardCharsets;
/** // 对数字和字节进行转换 假设数据存储是以大端模式存储的
* @Desc: ""
* @Author: caixiang
* @DATE: 2022/4/1 17:32
*/
//对数字和字节进行转换 假设数据存储是以大端模式存储的
// byte: 字节类型 占8位二进制 00000000 // byte: 字节类型 占8位二进制 00000000
// char: 字符类型 占2个字节 16位二进制 byte[0] byte[1] // char: 字符类型 占2个字节 16位二进制 byte[0] byte[1]
// int : 整数类型 占4个字节 32位二进制 byte[0] byte[1] byte[2] byte[3] // int : 整数类型 占4个字节 32位二进制 byte[0] byte[1] byte[2] byte[3]
@ -52,7 +47,7 @@ public class ByteUtil {
/** /**
* *
* 字符串转成16个字节的byte[] 不足16个字节前面填充0 * 字符串转成16个字节的byte[] 不足16个字节前面填充0
* */ */
public static byte[] stringTo16Byte(String input) { public static byte[] stringTo16Byte(String input) {
byte[] originalBytes = input.getBytes(StandardCharsets.UTF_8); byte[] originalBytes = input.getBytes(StandardCharsets.UTF_8);
ByteBuffer buffer = ByteBuffer.allocate(16); ByteBuffer buffer = ByteBuffer.allocate(16);
@ -84,7 +79,8 @@ public class ByteUtil {
if (ByteOrder.LITTLE_ENDIAN == byteOrder) { if (ByteOrder.LITTLE_ENDIAN == byteOrder) {
//小端模式数据的高字节保存在内存的高地址中而数据的低字节保存在内存的低地址中 //小端模式数据的高字节保存在内存的高地址中而数据的低字节保存在内存的低地址中
return (short) (bytes[0] & 0xff | (bytes[1] & 0xff) << Byte.SIZE); return (short) (bytes[0] & 0xff | (bytes[1] & 0xff) << Byte.SIZE);
} else { }
else {
return (short) (bytes[1] & 0xff | (bytes[0] & 0xff) << Byte.SIZE); return (short) (bytes[1] & 0xff | (bytes[0] & 0xff) << Byte.SIZE);
} }
} }
@ -113,7 +109,8 @@ public class ByteUtil {
if (ByteOrder.LITTLE_ENDIAN == byteOrder) { if (ByteOrder.LITTLE_ENDIAN == byteOrder) {
b[0] = (byte) (shortValue & 0xff); b[0] = (byte) (shortValue & 0xff);
b[1] = (byte) ((shortValue >> Byte.SIZE) & 0xff); b[1] = (byte) ((shortValue >> Byte.SIZE) & 0xff);
} else { }
else {
b[1] = (byte) (shortValue & 0xff); b[1] = (byte) (shortValue & 0xff);
b[0] = (byte) ((shortValue >> Byte.SIZE) & 0xff); b[0] = (byte) ((shortValue >> Byte.SIZE) & 0xff);
} }
@ -145,7 +142,8 @@ public class ByteUtil {
(bytes[1] & 0xFF) << 8 | // (bytes[1] & 0xFF) << 8 | //
(bytes[2] & 0xFF) << 16 | // (bytes[2] & 0xFF) << 16 | //
(bytes[3] & 0xFF) << 24; // (bytes[3] & 0xFF) << 24; //
} else { }
else {
return bytes[3] & 0xFF | // return bytes[3] & 0xFF | //
(bytes[2] & 0xFF) << 8 | // (bytes[2] & 0xFF) << 8 | //
(bytes[1] & 0xFF) << 16 | // (bytes[1] & 0xFF) << 16 | //
@ -183,7 +181,8 @@ public class ByteUtil {
(byte) ((intValue >> 24) & 0xFF) // (byte) ((intValue >> 24) & 0xFF) //
}; };
} else { }
else {
return new byte[]{ // return new byte[]{ //
(byte) ((intValue >> 24) & 0xFF), // (byte) ((intValue >> 24) & 0xFF), //
(byte) ((intValue >> 16) & 0xFF), // (byte) ((intValue >> 16) & 0xFF), //
@ -197,7 +196,8 @@ public class ByteUtil {
/** /**
* long转byte数组<br> * long转byte数组<br>
* 默认以小端序转换<br> * 默认以小端序转换<br>
* from: https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java * from:
* https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java
* *
* @param longValue long值 * @param longValue long值
* @return byte数组 * @return byte数组
@ -209,7 +209,8 @@ public class ByteUtil {
/** /**
* long转byte数组<br> * long转byte数组<br>
* 自定义端序<br> * 自定义端序<br>
* from: https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java * from:
* https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java
* *
* @param longValue long值 * @param longValue long值
* @param byteOrder 端序 * @param byteOrder 端序
@ -222,7 +223,8 @@ public class ByteUtil {
result[i] = (byte) (longValue & 0xFF); result[i] = (byte) (longValue & 0xFF);
longValue >>= Byte.SIZE; longValue >>= Byte.SIZE;
} }
} else { }
else {
for (int i = (result.length - 1); i >= 0; i--) { for (int i = (result.length - 1); i >= 0; i--) {
result[i] = (byte) (longValue & 0xFF); result[i] = (byte) (longValue & 0xFF);
longValue >>= Byte.SIZE; longValue >>= Byte.SIZE;
@ -234,7 +236,8 @@ public class ByteUtil {
/** /**
* byte数组转long<br> * byte数组转long<br>
* 默认以小端序转换<br> * 默认以小端序转换<br>
* from: https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java * from:
* https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java
* *
* @param bytes byte数组 * @param bytes byte数组
* @return long值 * @return long值
@ -246,7 +249,8 @@ public class ByteUtil {
/** /**
* byte数组转long<br> * byte数组转long<br>
* 自定义端序<br> * 自定义端序<br>
* from: https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java * from:
* https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java
* *
* @param bytes byte数组 * @param bytes byte数组
* @param byteOrder 端序 * @param byteOrder 端序
@ -259,7 +263,8 @@ public class ByteUtil {
values <<= Byte.SIZE; values <<= Byte.SIZE;
values |= (bytes[i] & 0xff); values |= (bytes[i] & 0xff);
} }
} else { }
else {
for (int i = 0; i < Long.BYTES; i++) { for (int i = 0; i < Long.BYTES; i++) {
values <<= Byte.SIZE; values <<= Byte.SIZE;
values |= (bytes[i] & 0xff); values |= (bytes[i] & 0xff);
@ -283,7 +288,8 @@ public class ByteUtil {
/** /**
* double转byte数组<br> * double转byte数组<br>
* 自定义端序<br> * 自定义端序<br>
* from: https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java * from:
* https://stackoverflow.com/questions/4485128/how-do-i-convert-long-to-byte-and-back-in-java
* *
* @param doubleValue double值 * @param doubleValue double值
* @param byteOrder 端序 * @param byteOrder 端序
@ -336,13 +342,17 @@ public class ByteUtil {
public static byte[] numberToBytes(Number number, ByteOrder byteOrder) { public static byte[] numberToBytes(Number number, ByteOrder byteOrder) {
if (number instanceof Double) { if (number instanceof Double) {
return doubleToBytes((Double) number, byteOrder); return doubleToBytes((Double) number, byteOrder);
} else if (number instanceof Long) { }
else if (number instanceof Long) {
return longToBytes((Long) number, byteOrder); return longToBytes((Long) number, byteOrder);
} else if (number instanceof Integer) { }
else if (number instanceof Integer) {
return intToBytes((Integer) number, byteOrder); return intToBytes((Integer) number, byteOrder);
} else if (number instanceof Short) { }
else if (number instanceof Short) {
return shortToBytes((Short) number, byteOrder); return shortToBytes((Short) number, byteOrder);
} else { }
else {
return doubleToBytes(number.doubleValue(), byteOrder); return doubleToBytes(number.doubleValue(), byteOrder);
} }
} }

View File

@ -1,20 +1,15 @@
package org.opentcs.kc.common.byteutils; package org.opentcs.kc.common.byteutils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2022/12/5 11:11
*/
public class CommonFunctions { public class CommonFunctions {
/** /**
* a 整除 b 如果有余数+1 * a 整除 b 如果有余数+1
* */ */
public static Integer exactDivision(Integer a,Integer b) { public static Integer exactDivision(Integer a, Integer b) {
int c = a/b; int c = a / b;
if(a%b!=0){ if (a % b != 0) {
c = a/b+1; c = a / b + 1;
} }
return c; return c;
} }

View File

@ -1,10 +1,5 @@
package org.opentcs.kc.common.byteutils; package org.opentcs.kc.common.byteutils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2024-05-08 11:02
*/
public class StringArrayStruc { public class StringArrayStruc {
private String[] content; private String[] content;
private Integer strSize; private Integer strSize;

View File

@ -11,7 +11,7 @@ public enum ModbusFC {
private Byte fread; private Byte fread;
private Byte fwrite; private Byte fwrite;
ModbusFC(Byte fread, Byte fwrite){ ModbusFC(Byte fread, Byte fwrite) {
this.fread = fread; this.fread = fread;
this.fwrite = fwrite; this.fwrite = fwrite;
} }

View File

@ -5,15 +5,13 @@ import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
/** /**
* @Author: 蔡翔 * 重点.
* @Date: 2019/11/6 14:11 * AsyncFuture 这个类就是票据 刚拿到这个票据是没有信息的当done == true 的时候这个票据 上就自动有信息了.
* @Version 1.0 * 这个结果类设计的比较神奇.
*
* 重点
* AsyncFuture 这个类就是票据 刚拿到这个票据是没有信息的当done == true 的时候这个票据 上就自动有信息了
* 这个结果类设计的比较神奇
*/ */
public class AsyncFuture<Object> implements Future<Object> { public class AsyncFuture<Object>
implements
Future<Object> {
private static final Logger logger = LoggerFactory.getLogger(AsyncFuture.class); private static final Logger logger = LoggerFactory.getLogger(AsyncFuture.class);
private volatile boolean done = false; private volatile boolean done = false;
@ -23,11 +21,12 @@ public class AsyncFuture<Object> implements Future<Object> {
public AsyncFuture(Object oldRequest) { public AsyncFuture(Object oldRequest) {
this.oldRequest = oldRequest; this.oldRequest = oldRequest;
} }
public AsyncFuture() { public AsyncFuture() {
} }
public void done(Object result){ public void done(Object result) {
synchronized (this){ synchronized (this) {
this.result = result; this.result = result;
this.done = true; this.done = true;
//注意这里的notifyAll只能唤醒 本锁的所有 下的所有 wait(),这里的锁就是 AsyncFuture这个类 //注意这里的notifyAll只能唤醒 本锁的所有 下的所有 wait(),这里的锁就是 AsyncFuture这个类
@ -37,13 +36,15 @@ public class AsyncFuture<Object> implements Future<Object> {
@Override @Override
public Object get(Long timeout) throws Exception { public Object get(Long timeout)
throws Exception {
return null; return null;
} }
@Override @Override
public Object get(Long timeout, String transationId) throws Exception { public Object get(Long timeout, String transationId)
synchronized (this){ throws Exception {
synchronized (this) {
//其实有 synchronize就相当于有一个阻塞队列当有线程执行了wait 方法就会把执行wait的这个线程给加入wait<Thread> 队列 //其实有 synchronize就相当于有一个阻塞队列当有线程执行了wait 方法就会把执行wait的这个线程给加入wait<Thread> 队列
//当有线程执行notify方法的时候就会往这个队列中取出一个或者多个Thread,取出来以后就能执行后续代码了 //当有线程执行notify方法的时候就会往这个队列中取出一个或者多个Thread,取出来以后就能执行后续代码了
// System.out.println("get"); // System.out.println("get");
@ -51,10 +52,10 @@ public class AsyncFuture<Object> implements Future<Object> {
// 当线程执行wait()会把当前的锁释放然后让出CPU进入等待状态 wait()会立刻释放synchronizedobj中的obj锁以便其他线程可以执行obj.notify() // 当线程执行wait()会把当前的锁释放然后让出CPU进入等待状态 wait()会立刻释放synchronizedobj中的obj锁以便其他线程可以执行obj.notify()
// * 当线程执行notify()/notifyAll()方法时会唤醒一个处于等待状态该对象锁的线程然后继续往下执行直到执行完退出对象锁锁住的区域synchronized修饰的代码块后再释放锁 // * 当线程执行notify()/notifyAll()方法时会唤醒一个处于等待状态该对象锁的线程然后继续往下执行直到执行完退出对象锁锁住的区域synchronized修饰的代码块后再释放锁
this.wait(timeout); this.wait(timeout);
if(!done){ if (!done) {
//logger.error("T3 timeout , request information: "+oldRequest.toString()); //logger.error("T3 timeout , request information: "+oldRequest.toString());
SendedList.remove(transationId); SendedList.remove(transationId);
throw new Exception("T3 timeout , request information: "+oldRequest.toString()); throw new Exception("T3 timeout , request information: " + oldRequest.toString());
} }
//因为上面的代码是加锁的所以这里的代码也是加锁的 //因为上面的代码是加锁的所以这里的代码也是加锁的

View File

@ -1,14 +1,11 @@
package org.opentcs.kc.syn; package org.opentcs.kc.syn;
/**
* @Author: 蔡翔
* @Date: 2019/11/6 13:49
* @Version 1.0
*/
public interface Future<MQMessage> { public interface Future<MQMessage> {
//别人调用我的时候我先给他们返回一个结果 //别人调用我的时候我先给他们返回一个结果
MQMessage get(Long timeout) throws Exception; MQMessage get(Long timeout)
throws Exception;
MQMessage get(Long timeout, String transationId) throws Exception; MQMessage get(Long timeout, String transationId)
throws Exception;
} }

View File

@ -1,13 +1,6 @@
package org.opentcs.kc.syn; package org.opentcs.kc.syn;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2021/8/17 14:35
*/
public class Main { public class Main {
// public static void main(String[] args) throws Exception { // public static void main(String[] args) throws Exception {

View File

@ -4,12 +4,6 @@ package org.opentcs.kc.syn;
import org.opentcs.kc.common.CaffeineUtil; import org.opentcs.kc.common.CaffeineUtil;
import org.opentcs.kc.common.Package; import org.opentcs.kc.common.Package;
/**
* @Desc: "MES端 发送远程指令列表"
* @Author: caixiang
* @DATE: 2021/8/17 14:14
*/
public class SendedList { public class SendedList {
// private static HashMap<String, AsyncFuture<MBPackage>> list = new HashMap<>(); // private static HashMap<String, AsyncFuture<MBPackage>> list = new HashMap<>();
// public static synchronized AsyncFuture<MQMessage> get(String transitionId){ // public static synchronized AsyncFuture<MQMessage> get(String transitionId){
@ -22,7 +16,7 @@ public class SendedList {
public static synchronized AsyncFuture<Package> add(String transitionId, Package PackageRequest) { public static synchronized AsyncFuture<Package> add(String transitionId, Package PackageRequest) {
AsyncFuture<Package> objectAsyncFuture = new AsyncFuture<>(PackageRequest); AsyncFuture<Package> objectAsyncFuture = new AsyncFuture<>(PackageRequest);
//list.put(transitionId,objectAsyncFuture); //list.put(transitionId,objectAsyncFuture);
CaffeineUtil.put(transitionId,objectAsyncFuture); CaffeineUtil.put(transitionId, objectAsyncFuture);
return objectAsyncFuture; return objectAsyncFuture;
} }
@ -35,8 +29,10 @@ public class SendedList {
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
public static synchronized Boolean set(String transitionId, Package message) { public static synchronized Boolean set(String transitionId, Package message) {
//AsyncFuture<MBPackage> mqMessageAsyncFuture = list.get(transitionId); //AsyncFuture<MBPackage> mqMessageAsyncFuture = list.get(transitionId);
AsyncFuture<Package> mqMessageAsyncFuture = (AsyncFuture<Package>)CaffeineUtil.get(transitionId); AsyncFuture<Package> mqMessageAsyncFuture = (AsyncFuture<Package>) CaffeineUtil.get(
if(mqMessageAsyncFuture == null){ transitionId
);
if (mqMessageAsyncFuture == null) {
return false; return false;
} }
mqMessageAsyncFuture.done(message); mqMessageAsyncFuture.done(message);
@ -46,6 +42,4 @@ public class SendedList {
} }
} }

View File

@ -20,7 +20,7 @@ public class LogConst {
"离别的风,风干了月的泪。夜里的美", "离别的风,风干了月的泪。夜里的美",
"是梦的呢喃低语,挥走一片彩云,段落成珠。拂袖离去,乘鹤而来,古道西风瘦马。斑驳的树影中,眉目如画的眼,轻语告别了往事如烟。", "是梦的呢喃低语,挥走一片彩云,段落成珠。拂袖离去,乘鹤而来,古道西风瘦马。斑驳的树影中,眉目如画的眼,轻语告别了往事如烟。",
"无言的殇,几世沧桑,几生悲凉。一起剪了西窗烛,听了夜来风吹雨。昨日的叹息,今日的迷离,执一伞,存了一世一笔的温情。一曲长歌,唱尽了一世繁华,一世缘……", "无言的殇,几世沧桑,几生悲凉。一起剪了西窗烛,听了夜来风吹雨。昨日的叹息,今日的迷离,执一伞,存了一世一笔的温情。一曲长歌,唱尽了一世繁华,一世缘……",
"一世恋书,那便十里花开。一生凄凉,那便霜花十里。" , "一世恋书,那便十里花开。一生凄凉,那便霜花十里。",
"一抹浓烟,便翻页书,展颜一笑,是时间带来遥远的梦。细数树的年轮,感受昨日惆怅,留一半清醒,梦一半叶落。在指尖流过的沙,海边浪花一朵朵,不相遇,才有不约而同。", "一抹浓烟,便翻页书,展颜一笑,是时间带来遥远的梦。细数树的年轮,感受昨日惆怅,留一半清醒,梦一半叶落。在指尖流过的沙,海边浪花一朵朵,不相遇,才有不约而同。",
"这世俗,太多牵挂留在心间,一点朱砂泪,一曲相诗歌。岁月朦胧,梦醒了人生,风雨相容,演绎了一段风情。雪亦梦,雨亦梦,万张红纸从天洒来。惊动了山,撼动了天。" + "这世俗,太多牵挂留在心间,一点朱砂泪,一曲相诗歌。岁月朦胧,梦醒了人生,风雨相容,演绎了一段风情。雪亦梦,雨亦梦,万张红纸从天洒来。惊动了山,撼动了天。" +
"一纸情愁,一指烟凉。一相思,一思量,水漫岸头,我们都有着自己不同的三生故事。迎一夜秋风,送一世暖阳,一切冰雪里的花开,是我一生的柔情。" + "一纸情愁,一指烟凉。一相思,一思量,水漫岸头,我们都有着自己不同的三生故事。迎一夜秋风,送一世暖阳,一切冰雪里的花开,是我一生的柔情。" +
@ -29,7 +29,8 @@ public class LogConst {
"这便,晨光微好,花开静好……"}; "这便,晨光微好,花开静好……"};
private final static Random r = new Random(); private final static Random r = new Random();
public static String getLogInfo(){
return LOG_INFOS[r.nextInt(LOG_INFOS.length-1)]; public static String getLogInfo() {
return LOG_INFOS[r.nextInt(LOG_INFOS.length - 1)];
} }
} }

View File

@ -32,18 +32,24 @@ public class LogEventBroadcaster {
.handler(new LogEventEncoder(remoteAddress)); .handler(new LogEventEncoder(remoteAddress));
} }
public void run() throws Exception { public void run()
throws Exception {
//绑定 Channel //绑定 Channel
Channel ch = bootstrap.bind(0).sync().channel(); Channel ch = bootstrap.bind(0).sync().channel();
long count = 0; long count = 0;
//启动主处理循环模拟日志发送 //启动主处理循环模拟日志发送
for (;;) { for (;;) {
ch.writeAndFlush(new LogMsg(null, ++count, ch.writeAndFlush(
LogConst.getLogInfo())); new LogMsg(
null, ++count,
LogConst.getLogInfo()
)
);
try { try {
//休眠 2 如果被中断则退出循环 //休眠 2 如果被中断则退出循环
Thread.sleep(2000); Thread.sleep(2000);
} catch (InterruptedException e) { }
catch (InterruptedException e) {
Thread.interrupted(); Thread.interrupted();
break; break;
} }
@ -54,15 +60,19 @@ public class LogEventBroadcaster {
group.shutdownGracefully(); group.shutdownGracefully();
} }
public static void main(String[] args) throws Exception { public static void main(String[] args)
throws Exception {
//创建并启动一个新的 UdpQuestionSide 的实例 //创建并启动一个新的 UdpQuestionSide 的实例
LogEventBroadcaster broadcaster = new LogEventBroadcaster( LogEventBroadcaster broadcaster = new LogEventBroadcaster(
//表明本应用发送的报文并没有一个确定的目的地也就是进行广播 //表明本应用发送的报文并没有一个确定的目的地也就是进行广播
//就是往这个网络下 所有主机发送数据往这些主机的 LogConst.MONITOR_SIDE_PORT端口 发送数据 //就是往这个网络下 所有主机发送数据往这些主机的 LogConst.MONITOR_SIDE_PORT端口 发送数据
//todo 这里的设置和 单播是有差异的 //todo 这里的设置和 单播是有差异的
new InetSocketAddress("255.255.255.255", new InetSocketAddress(
LogConst.MONITOR_SIDE_PORT)); "255.255.255.255",
LogConst.MONITOR_SIDE_PORT
)
);
try { try {
System.out.println("广播服务启动"); System.out.println("广播服务启动");
broadcaster.run(); broadcaster.run();

View File

@ -18,17 +18,21 @@ import java.util.List;
* 作者DarkKIng * 作者DarkKIng
* 类说明解码将DatagramPacket解码为实际的日志实体类 * 类说明解码将DatagramPacket解码为实际的日志实体类
*/ */
public class LogEventDecoder extends MessageToMessageDecoder<DatagramPacket> { public class LogEventDecoder
extends
MessageToMessageDecoder<DatagramPacket> {
@Override @Override
protected void decode(ChannelHandlerContext ctx, protected void decode(
DatagramPacket datagramPacket, List<Object> out) ChannelHandlerContext ctx,
DatagramPacket datagramPacket, List<Object> out
)
throws Exception { throws Exception {
//获取对 DatagramPacket 中的数据ByteBuf的引用 //获取对 DatagramPacket 中的数据ByteBuf的引用
ByteBuf data = datagramPacket.content(); ByteBuf data = datagramPacket.content();
long time = new Date().getTime(); long time = new Date().getTime();
System.out.println(time+" 接受到发送的消息:"); System.out.println(time + " 接受到发送的消息:");
//获得消息的id //获得消息的id
long msgId = data.readLong(); long msgId = data.readLong();
//获得分隔符SEPARATOR //获得分隔符SEPARATOR
@ -36,11 +40,15 @@ public class LogEventDecoder extends MessageToMessageDecoder<DatagramPacket> {
//获取读索引的当前位置就是分隔符的索引+1 //获取读索引的当前位置就是分隔符的索引+1
int idx = data.readerIndex(); int idx = data.readerIndex();
//提取日志消息从读索引开始到最后为日志的信息 //提取日志消息从读索引开始到最后为日志的信息
String sendMsg = data.slice(idx , String sendMsg = data.slice(
data.readableBytes()).toString(CharsetUtil.UTF_8); idx,
data.readableBytes()
).toString(CharsetUtil.UTF_8);
//构建一个新的 LogMsg 对象并且将它添加到已经解码的消息的列表中 //构建一个新的 LogMsg 对象并且将它添加到已经解码的消息的列表中
LogMsg event = new LogMsg(datagramPacket.sender(), LogMsg event = new LogMsg(
msgId, sendMsg); datagramPacket.sender(),
msgId, sendMsg
);
//作为本handler的处理结果交给后面的handler进行处理 //作为本handler的处理结果交给后面的handler进行处理
out.add(event); out.add(event);
} }

View File

@ -18,7 +18,9 @@ import java.util.List;
* 作者DarkKIng * 作者DarkKIng
* 类说明编码将实际的日志实体类编码为DatagramPacket * 类说明编码将实际的日志实体类编码为DatagramPacket
*/ */
public class LogEventEncoder extends MessageToMessageEncoder<LogMsg> { public class LogEventEncoder
extends
MessageToMessageEncoder<LogMsg> {
private final InetSocketAddress remoteAddress; private final InetSocketAddress remoteAddress;
//LogEventEncoder 创建了即将被发送到指定的 InetSocketAddress //LogEventEncoder 创建了即将被发送到指定的 InetSocketAddress
@ -28,12 +30,15 @@ public class LogEventEncoder extends MessageToMessageEncoder<LogMsg> {
} }
@Override @Override
protected void encode(ChannelHandlerContext channelHandlerContext, protected void encode(
LogMsg logMsg, List<Object> out) throws Exception { ChannelHandlerContext channelHandlerContext,
LogMsg logMsg, List<Object> out
)
throws Exception {
byte[] msg = logMsg.getMsg().getBytes(CharsetUtil.UTF_8); byte[] msg = logMsg.getMsg().getBytes(CharsetUtil.UTF_8);
//容量的计算两个long型+消息的内容+分割符 //容量的计算两个long型+消息的内容+分割符
ByteBuf buf = channelHandlerContext.alloc() ByteBuf buf = channelHandlerContext.alloc()
.buffer(8*2 + msg.length + 1); .buffer(8 * 2 + msg.length + 1);
//将发送时间写入到 ByteBuf中 //将发送时间写入到 ByteBuf中
buf.writeLong(logMsg.getTime()); buf.writeLong(logMsg.getTime());
//将消息id写入到 ByteBuf中 //将消息id写入到 ByteBuf中

View File

@ -14,19 +14,26 @@ import io.netty.channel.SimpleChannelInboundHandler;
* 类说明日志的业务处理类,实际的业务处理接受日志信息 * 类说明日志的业务处理类,实际的业务处理接受日志信息
*/ */
public class LogEventHandler public class LogEventHandler
extends SimpleChannelInboundHandler<LogMsg> { extends
SimpleChannelInboundHandler<LogMsg> {
@Override @Override
public void exceptionCaught(ChannelHandlerContext ctx, public void exceptionCaught(
Throwable cause) throws Exception { ChannelHandlerContext ctx,
Throwable cause
)
throws Exception {
//当异常发生时打印栈跟踪信息并关闭对应的 Channel //当异常发生时打印栈跟踪信息并关闭对应的 Channel
cause.printStackTrace(); cause.printStackTrace();
ctx.close(); ctx.close();
} }
@Override @Override
public void channelRead0(ChannelHandlerContext ctx, public void channelRead0(
LogMsg event) throws Exception { ChannelHandlerContext ctx,
LogMsg event
)
throws Exception {
//创建 StringBuilder并且构建输出的字符串 //创建 StringBuilder并且构建输出的字符串
StringBuilder builder = new StringBuilder(); StringBuilder builder = new StringBuilder();
builder.append(event.getTime()); builder.append(event.getTime());

View File

@ -6,12 +6,6 @@ import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioDatagramChannel; import io.netty.channel.socket.nio.NioDatagramChannel;
import java.net.InetSocketAddress; import java.net.InetSocketAddress;
/**
* @Desc: ""
* @Author: caixiang ( Server 接受消息 )
* @DATE: 2024/12/15 10:57
*/
public class LogEventMonitor { public class LogEventMonitor {
private final EventLoopGroup group; private final EventLoopGroup group;
private final Bootstrap bootstrap; private final Bootstrap bootstrap;
@ -25,8 +19,8 @@ public class LogEventMonitor {
//设置套接字选项 SO_BROADCAST //设置套接字选项 SO_BROADCAST
.option(ChannelOption.SO_BROADCAST, true) .option(ChannelOption.SO_BROADCAST, true)
//允许重用 //允许重用
.option(ChannelOption.SO_REUSEADDR,true) .option(ChannelOption.SO_REUSEADDR, true)
.handler( new ChannelInitializer<Channel>() { .handler(new ChannelInitializer<Channel>() {
@Override @Override
protected void initChannel(Channel channel) protected void initChannel(Channel channel)
throws Exception { throws Exception {
@ -34,7 +28,7 @@ public class LogEventMonitor {
pipeline.addLast(new LogEventDecoder()); pipeline.addLast(new LogEventDecoder());
pipeline.addLast(new LogEventHandler()); pipeline.addLast(new LogEventHandler());
} }
} ) })
.localAddress(address); .localAddress(address);
} }
@ -47,16 +41,19 @@ public class LogEventMonitor {
group.shutdownGracefully(); group.shutdownGracefully();
} }
public static void main(String[] args) throws Exception { public static void main(String[] args)
throws Exception {
//构造一个新的 UdpAnswerSide并指明监听端口 //构造一个新的 UdpAnswerSide并指明监听端口
LogEventMonitor monitor = new LogEventMonitor( LogEventMonitor monitor = new LogEventMonitor(
new InetSocketAddress(LogConst.MONITOR_SIDE_PORT)); new InetSocketAddress(LogConst.MONITOR_SIDE_PORT)
);
try { try {
//绑定本地监听端口 //绑定本地监听端口
Channel channel = monitor.bind(); Channel channel = monitor.bind();
System.out.println("UdpAnswerSide running"); System.out.println("UdpAnswerSide running");
channel.closeFuture().sync(); channel.closeFuture().sync();
} finally { }
finally {
monitor.stop(); monitor.stop();
} }
} }

View File

@ -14,24 +14,26 @@ import java.net.InetSocketAddress;
*/ */
public final class LogMsg { public final class LogMsg {
public static final byte SEPARATOR = (byte) ':'; public static final byte SEPARATOR = (byte) ':';
/*源的 InetSocketAddress*/ /* 源的 InetSocketAddress */
private final InetSocketAddress source; private final InetSocketAddress source;
/*消息内容*/ /* 消息内容 */
private final String msg; private final String msg;
/*消息id*/ /* 消息id */
private final long msgId; private final long msgId;
/*消息发送的时间*/ /* 消息发送的时间 */
private final long time; private final long time;
//用于传入消息的构造函数 //用于传入消息的构造函数
public LogMsg(String msg) { public LogMsg(String msg) {
this(null, msg,-1,System.currentTimeMillis()); this(null, msg, -1, System.currentTimeMillis());
} }
//用于传出消息的构造函数 //用于传出消息的构造函数
public LogMsg(InetSocketAddress source, long msgId, public LogMsg(
String msg) { InetSocketAddress source, long msgId,
this(source,msg,msgId,System.currentTimeMillis()); String msg
) {
this(source, msg, msgId, System.currentTimeMillis());
} }
public LogMsg(InetSocketAddress source, String msg, long msgId, long time) { public LogMsg(InetSocketAddress source, String msg, long msgId, long time) {

View File

@ -3,12 +3,13 @@ package org.opentcs.kc.udp;
import java.nio.ByteOrder; import java.nio.ByteOrder;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
//import org.opentcs.kc.udp.agv.param.AgvEvent;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
import org.opentcs.kc.udp.agv.param.AgvEvent; import org.opentcs.kc.udp.agv.param.AgvEvent;
import org.opentcs.kc.udp.agv.param.AgvEventConstant; import org.opentcs.kc.udp.agv.param.AgvEventConstant;
import org.opentcs.kc.udp.agv.param.function.af.QueryRobotStatusRsp;
import org.opentcs.kc.udp.agv.param.function.b1.SubscribeInfo; import org.opentcs.kc.udp.agv.param.function.b1.SubscribeInfo;
import org.opentcs.kc.udp.agv.param.function.b1.SubscribeParam; import org.opentcs.kc.udp.agv.param.function.b1.SubscribeParam;
import org.opentcs.kc.udp.agv.param.function.b1.SubscribeRsp;
import org.opentcs.kc.udp.agv.param.function.navigation.*; import org.opentcs.kc.udp.agv.param.function.navigation.*;
import org.opentcs.kc.udp.agv.param.function.read.ReadParam; import org.opentcs.kc.udp.agv.param.function.read.ReadParam;
import org.opentcs.kc.udp.agv.param.function.read.ReadStrValue; import org.opentcs.kc.udp.agv.param.function.read.ReadStrValue;
@ -17,12 +18,11 @@ import org.opentcs.kc.udp.agv.param.function.write.WriteParam;
import org.opentcs.kc.udp.agv.param.function.write.WriteStrValue; import org.opentcs.kc.udp.agv.param.function.write.WriteStrValue;
import org.opentcs.kc.udp.agv.param.function.write.WriteValueMember; import org.opentcs.kc.udp.agv.param.function.write.WriteValueMember;
import org.opentcs.kc.udp.agv.param.function.x14.RobotSetPosition; import org.opentcs.kc.udp.agv.param.function.x14.RobotSetPosition;
import org.opentcs.kc.udp.agv.param.function.x17.QueryRobotRunStatusRsp;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
import org.opentcs.kc.udp.agv.param.AgvEvent;
/** /**
*
* AGV启动 * AGV启动
* 0xAF(查询机器人状态) 👌 * 0xAF(查询机器人状态) 👌
* 0xB1(下发订阅信令) 👌 * 0xB1(下发订阅信令) 👌
@ -33,26 +33,27 @@ import org.opentcs.kc.udp.agv.param.AgvEvent;
* 0x1F(确认初始位置) 👌 * 0x1F(确认初始位置) 👌
* 运行 * 运行
* 0xAE(导航控制导航点控制) 👌 * 0xAE(导航控制导航点控制) 👌
*
*
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/17 16:25
*/ */
public class KCCommandDemo { public class KCCommandDemo {
public static void main(String [] args) throws Exception{ public static void main(String[] args)
throws Exception {
// { // {
// //0xAF(查询机器人状态) // //0xAF(查询机器人状态)
// AgvEvent agvEvent = queryStatus(); // AgvEvent agvEvent = queryStatus();
// printInfo(agvEvent); // printInfo(agvEvent);
// RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); // RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
// if(rcv.isOk()){ // if(rcv.isOk()){
// QueryRobotStatusRsp queryRobotStatusRsp = new QueryRobotStatusRsp(rcv.getDataBytes());
// System.out.println(); // System.out.println();
// System.out.println("received transationId : "+ "isok:"+rcv.isOk()); // System.out.println("received transationId : "+ "isok:"+rcv.isOk());
// for (byte b:rcv.getValue()){ // for (byte b:rcv.getValue()){
// System.out.print(byteToHex(b)+" "); // System.out.print(byteToHex(b)+" ");
// } // }
// System.out.println();
// System.out.println("---------------------");
// for (byte c:rcv.getDataBytes()){
// System.out.print(byteToHex(c)+" ");
// }
// QueryRobotStatusRsp queryRobotStatusRsp = new QueryRobotStatusRsp(rcv.getDataBytes());
// }else { // }else {
// System.out.println(); // System.out.println();
// System.out.println("received transationId : "+ "isok:"+rcv.isOk()); // System.out.println("received transationId : "+ "isok:"+rcv.isOk());
@ -86,6 +87,10 @@ public class KCCommandDemo {
// if(rcv.isOk()){ // if(rcv.isOk()){
// System.out.println(); // System.out.println();
// System.out.println("received transationId : "+ "isok:"+rcv.isOk()); // System.out.println("received transationId : "+ "isok:"+rcv.isOk());
// for (byte b:rcv.getDataBytes()) {
// System.out.print(byteToHex(b)+" ");
// }
//
// SubscribeRsp subscribeRsp = new SubscribeRsp(rcv.getDataBytes()); // SubscribeRsp subscribeRsp = new SubscribeRsp(rcv.getDataBytes());
// if(subscribeRsp.isOk()){ // if(subscribeRsp.isOk()){
// //... // //...
@ -142,26 +147,26 @@ public class KCCommandDemo {
// } // }
// { {
// //0x14(手动定位) //0x14(手动定位)
// AgvEvent agvEvent = manualLocation(); AgvEvent agvEvent = manualLocation();
// printInfo(agvEvent); printInfo(agvEvent);
// RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
// if(rcv.isOk()){ if(rcv.isOk()){
// System.out.println(); System.out.println();
// System.out.println("received "+ "isok:"+rcv.isOk()+" dataBytes:"); System.out.println("received "+ "isok:"+rcv.isOk()+" dataBytes:");
// printInfo(rcv); printInfo(rcv);
// if(rcv.isOk()){ if(rcv.isOk()){
// //get and parse value //get and parse value
// System.out.println("0x14 ok"); System.out.println("0x14 ok");
// }else { }else {
// System.out.println("0x14 failed"); System.out.println("0x14 failed");
// } }
// }else { }else {
// System.out.println(); System.out.println();
// System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : "+ "isok:"+rcv.isOk());
// } }
// } }
// { // {
@ -170,33 +175,40 @@ public class KCCommandDemo {
// printInfo(agvEvent); // printInfo(agvEvent);
// RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); // RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
// if(rcv.isOk()){ // if(rcv.isOk()){
// //QueryCargoStatusRsp queryCargoStatusRsp = new QueryCargoStatusRsp(rcv.getDataBytes());
// QueryRobotRunStatusRsp queryRobotRunStatusRsp = new QueryRobotRunStatusRsp(rcv.getDataBytes());
// System.out.println(queryRobotRunStatusRsp.toString());
// System.out.println(); // System.out.println();
// System.out.println("received transationId : "+ "isok:"+rcv.isOk()); // System.out.println("received transationId : "+ "isok:"+rcv.isOk());
// for (byte b:rcv.getValue()){ // for (byte b:rcv.getValue()){
// System.out.print(byteToHex(b)+" "); // System.out.print(byteToHex(b)+" ");
// } // }
// System.out.println();
// System.out.println("print databytes:-----=======");
// for (byte c:rcv.getDataBytes()){
// System.out.print(byteToHex(c)+" ");
// }
// //QueryCargoStatusRsp queryCargoStatusRsp = new QueryCargoStatusRsp(rcv.getDataBytes());
// QueryRobotRunStatusRsp queryRobotRunStatusRsp = new QueryRobotRunStatusRsp(rcv.getDataBytes());
// System.out.println(queryRobotRunStatusRsp.toString());
//
// }else { // }else {
// System.out.println(); // System.out.println();
// System.out.println("received transationId : "+ "isok:"+rcv.isOk()); // System.out.println("received transationId : "+ "isok:"+rcv.isOk());
// } // }
// } // }
{ // {
//0x1F(确认初始位置) // //0x1F(确认初始位置)
AgvEvent agvEvent = confirmInitialPosition(); // AgvEvent agvEvent = confirmInitialPosition();
printInfo(agvEvent); // printInfo(agvEvent);
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); // RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ // if (rcv.isOk()) {
System.out.println("0x1F ok"); // System.out.println("0x1F ok");
}else { // }
System.out.println(); // else {
System.out.println("0x1F fail"); // System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); // System.out.println("0x1F fail");
} // System.out.println("received transationId : " + "isok:" + rcv.isOk());
} // }
// }
// { // {
// //0xAE(导航控制导航点控制) // //0xAE(导航控制导航点控制)
@ -227,17 +239,17 @@ public class KCCommandDemo {
return hexString; return hexString;
} }
public static void printInfo(AgvEvent agvEvent){ public static void printInfo(AgvEvent agvEvent) {
System.out.println("sended transationId : "+agvEvent.getTransationIdString()); System.out.println("sended transationId : " + agvEvent.getTransationIdString());
for (byte b:agvEvent.toBytes().getBody()){ for (byte b : agvEvent.toBytes().getBody()) {
System.out.print(byteToHex(b)+" "); System.out.print(byteToHex(b) + " ");
} }
} }
public static void printInfo(RcvEventPackage rcv){ public static void printInfo(RcvEventPackage rcv) {
for (byte b: rcv.getDataBytes()){ for (byte b : rcv.getDataBytes()) {
System.out.print(byteToHex(b)+" "); System.out.print(byteToHex(b) + " ");
} }
} }
@ -246,18 +258,18 @@ public class KCCommandDemo {
* 指令0x02 * 指令0x02
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent readValue() { public static AgvEvent readValue() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_READ); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_READ);
List<ReadValueMember> readValueMemberList = new ArrayList<>(); List<ReadValueMember> readValueMemberList = new ArrayList<>();
ReadValueMember readValueMember1 = new ReadValueMember(Short.valueOf("0"),Short.valueOf("1")); ReadValueMember readValueMember1 = new ReadValueMember(Short.valueOf("0"), Short.valueOf("1"));
readValueMemberList.add(readValueMember1); readValueMemberList.add(readValueMember1);
List<ReadStrValue> readStrValueList = new ArrayList<>(); List<ReadStrValue> readStrValueList = new ArrayList<>();
ReadStrValue readStrValue = new ReadStrValue("Battry_SOC", 1, readValueMemberList); ReadStrValue readStrValue = new ReadStrValue("Battry_SOC", 1, readValueMemberList);
readStrValueList.add(readStrValue); readStrValueList.add(readStrValue);
ReadParam readParam = new ReadParam(agvEvent.getTransationId(),readStrValueList ); ReadParam readParam = new ReadParam(agvEvent.getTransationId(), readStrValueList);
agvEvent.setBody(readParam.toBytes()); agvEvent.setBody(readParam.toBytes());
return agvEvent; return agvEvent;
@ -268,33 +280,33 @@ public class KCCommandDemo {
* 指令0x03 * 指令0x03
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent writeValue() { public static AgvEvent writeValue() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_WRITE); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_WRITE);
List<WriteValueMember> valueMemberList = new ArrayList<>(); List<WriteValueMember> valueMemberList = new ArrayList<>();
WriteValueMember valueMember1 = new WriteValueMember(Short.valueOf("0"),Short.valueOf("4"), ByteUtils.uintToBytes(3, ByteOrder.LITTLE_ENDIAN)); WriteValueMember valueMember1 = new WriteValueMember(
Short.valueOf("0"), Short.valueOf("4"), ByteUtils.uintToBytes(3, ByteOrder.LITTLE_ENDIAN)
);
valueMemberList.add(valueMember1); valueMemberList.add(valueMember1);
List<WriteStrValue> strValueList = new ArrayList<>(); List<WriteStrValue> strValueList = new ArrayList<>();
WriteStrValue strValue = new WriteStrValue("TestRW", 1, valueMemberList); WriteStrValue strValue = new WriteStrValue("TestRW", 1, valueMemberList);
strValueList.add(strValue); strValueList.add(strValue);
WriteParam param = new WriteParam(1,strValueList ); WriteParam param = new WriteParam(1, strValueList);
agvEvent.setBody(param.toBytes()); agvEvent.setBody(param.toBytes());
return agvEvent; return agvEvent;
} }
/** /**
* decs: 查询机器人状态 * decs: 查询机器人状态
* 指令0xAF * 指令0xAF
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent queryStatus() { public static AgvEvent queryStatus() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_STATUS); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_STATUS);
return agvEvent; return agvEvent;
@ -305,7 +317,7 @@ public class KCCommandDemo {
* 指令0xAE * 指令0xAE
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent navigationControl() { public static AgvEvent navigationControl() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_STATUS); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_STATUS);
//TODO 构建 //TODO 构建
@ -313,34 +325,69 @@ public class KCCommandDemo {
//构建point //构建point
Action[] pointActions1 = new Action[]{ Action[] pointActions1 = new Action[]{
new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01)) new Action(
ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(
orderId, (byte) 0x01
)
)
//,new Action()... 每一个point 可以绑定一个或者多个 action //,new Action()... 每一个point 可以绑定一个或者多个 action
}; };
Action[] pointActions2 = new Action[]{ Action[] pointActions2 = new Action[]{
new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01)) new Action(
ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(
orderId, (byte) 0x01
)
)
}; };
Action[] pointActions3 = new Action[]{ Action[] pointActions3 = new Action[]{
new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01)) new Action(
ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(
orderId, (byte) 0x01
)
)
}; };
Point[] points = new Point[]{ Point[] points = new Point[]{
new Point(0, 1, 1f, (byte)0x00, ByteUtils.usintTo1Byte(pointActions1.length),pointActions1), new Point(
new Point(2, 2, 1f, (byte)0x00, ByteUtils.usintTo1Byte(pointActions2.length),pointActions2), 0, 1, 1f, (byte) 0x00, ByteUtils.usintTo1Byte(pointActions1.length), pointActions1
new Point(4, 3, 1f, (byte)0x00, ByteUtils.usintTo1Byte(pointActions3.length),pointActions3) ),
new Point(
2, 2, 1f, (byte) 0x00, ByteUtils.usintTo1Byte(pointActions2.length), pointActions2
),
new Point(
4, 3, 1f, (byte) 0x00, ByteUtils.usintTo1Byte(pointActions3.length), pointActions3
)
}; };
//构建path //构建path
Action[] pathActions1 = new Action[]{ Action[] pathActions1 = new Action[]{
new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01)) new Action(
ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(
orderId, (byte) 0x01
)
)
//,new Action()... 每一个path 可以绑定一个或者多个 action //,new Action()... 每一个path 可以绑定一个或者多个 action
}; };
Action[] pathActions2 = new Action[]{ Action[] pathActions2 = new Action[]{
new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01)) new Action(
ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(
orderId, (byte) 0x01
)
)
}; };
Path[] paths = new Path[]{ Path[] paths = new Path[]{
new Path(1,1,1f,(byte)0x00,(byte)0x00,ByteUtils.usintTo1Byte(pathActions1.length),5f,1f,pathActions1) , new Path(
new Path(3,2,1f,(byte)0x00,(byte)0x00,ByteUtils.usintTo1Byte(pathActions2.length),5f,1f,pathActions2) , 1, 1, 1f, (byte) 0x00, (byte) 0x00, ByteUtils.usintTo1Byte(pathActions1.length), 5f, 1f,
pathActions1
),
new Path(
3, 2, 1f, (byte) 0x00, (byte) 0x00, ByteUtils.usintTo1Byte(pathActions2.length), 5f, 1f,
pathActions2
),
}; };
NavigationParam navigationParam = new NavigationParam(1,1,ByteUtils.usintTo1Byte(points.length),ByteUtils.usintTo1Byte(points.length-1),points,paths); NavigationParam navigationParam = new NavigationParam(
1, 1, ByteUtils.usintTo1Byte(points.length), ByteUtils.usintTo1Byte(points.length - 1),
points, paths
);
agvEvent.setBody(navigationParam.toBytes()); agvEvent.setBody(navigationParam.toBytes());
return agvEvent; return agvEvent;
@ -351,7 +398,7 @@ public class KCCommandDemo {
* 指令0x1F * 指令0x1F
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent confirmInitialPosition() { public static AgvEvent confirmInitialPosition() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_CONFIRM_ROBOT_POSITION); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_CONFIRM_ROBOT_POSITION);
return agvEvent; return agvEvent;
@ -362,7 +409,7 @@ public class KCCommandDemo {
* 指令0x17 * 指令0x17
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent queryRobotRunStatus() { public static AgvEvent queryRobotRunStatus() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_RUN_STATUS); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_RUN_STATUS);
return agvEvent; return agvEvent;
@ -382,7 +429,7 @@ public class KCCommandDemo {
* 指令0xB0 * 指令0xB0
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent checkCargoStatus() { public static AgvEvent checkCargoStatus() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_CARRY_STATUS); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_CARRY_STATUS);
return agvEvent; return agvEvent;
@ -393,14 +440,18 @@ public class KCCommandDemo {
* 指令0xB1 * 指令0xB1
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent issueSubscribe() { public static AgvEvent issueSubscribe() {
List<SubscribeInfo> subscribeInfoList = new ArrayList<>(); List<SubscribeInfo> subscribeInfoList = new ArrayList<>();
SubscribeInfo subscribeInfo = new SubscribeInfo(new byte[]{(byte)0xaf,(byte)0x00}, (short) 100,1000); SubscribeInfo subscribeInfo = new SubscribeInfo(
new byte[]{(byte) 0xaf, (byte) 0x00}, (short) 100, 1000
);
//SubscribeInfo subscribeInfo = new SubscribeInfo(new byte[]{(byte)0xb0,(byte)0x00}, (short) 100,1000); //SubscribeInfo subscribeInfo = new SubscribeInfo(new byte[]{(byte)0xb0,(byte)0x00}, (short) 100,1000);
subscribeInfoList.add(subscribeInfo); subscribeInfoList.add(subscribeInfo);
SubscribeParam subscribeParam = new SubscribeParam(subscribeInfoList); SubscribeParam subscribeParam = new SubscribeParam(subscribeInfoList);
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_ISSUE_SUBSCRIPTION,subscribeParam.toBytes()); AgvEvent agvEvent = new AgvEvent(
AgvEventConstant.CommandCode_ISSUE_SUBSCRIPTION, subscribeParam.toBytes()
);
return agvEvent; return agvEvent;
} }

View File

@ -3,11 +3,6 @@ package org.opentcs.kc.udp.Service;
import org.opentcs.kc.udp.agv.param.AgvEvent; import org.opentcs.kc.udp.agv.param.AgvEvent;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
/**
* @author xzh
* @date 2025/2/21
* @desc 命令基类
*/
public class BaseCommand { public class BaseCommand {
public static String byteToHex(byte b) { public static String byteToHex(byte b) {
@ -22,17 +17,17 @@ public class BaseCommand {
return hexString; return hexString;
} }
public static void printInfo(AgvEvent agvEvent){ public static void printInfo(AgvEvent agvEvent) {
System.out.println("sended transationId : "+agvEvent.getTransationIdString()); System.out.println("sended transationId : " + agvEvent.getTransationIdString());
for (byte b:agvEvent.toBytes().getBody()){ for (byte b : agvEvent.toBytes().getBody()) {
System.out.print(byteToHex(b)+" "); System.out.print(byteToHex(b) + " ");
} }
} }
public static void printInfo(RcvEventPackage rcv){ public static void printInfo(RcvEventPackage rcv) {
for (byte b: rcv.getDataBytes()){ for (byte b : rcv.getDataBytes()) {
System.out.print(byteToHex(b)+" "); System.out.print(byteToHex(b) + " ");
} }
} }

View File

@ -5,19 +5,16 @@ import org.opentcs.kc.udp.agv.param.AgvEventConstant;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
/** public class ConfirmRelocation
* @author xzh extends
* @date 2025/2/21 BaseCommand {
* @desc 确认位置 0x1F
*/
public class ConfirmRelocation extends BaseCommand{
/** /**
* decs: 确认机器人位置 * decs: 确认机器人位置.
* 指令0x1F * 指令0x1F.
* author: caixiang * author: caixiang.
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25.
* */ */
public static AgvEvent confirmInitialPosition() { public static AgvEvent confirmInitialPosition() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_CONFIRM_ROBOT_POSITION); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_CONFIRM_ROBOT_POSITION);
return agvEvent; return agvEvent;
@ -28,12 +25,13 @@ public class ConfirmRelocation extends BaseCommand{
AgvEvent agvEvent = confirmInitialPosition(); AgvEvent agvEvent = confirmInitialPosition();
printInfo(agvEvent); printInfo(agvEvent);
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ if (rcv.isOk()) {
System.out.println("0x1F ok"); System.out.println("0x1F ok");
}else { }
else {
System.out.println(); System.out.println();
System.out.println("0x1F fail"); System.out.println("0x1F fail");
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
} }
} }
} }

View File

@ -1,5 +1,7 @@
package org.opentcs.kc.udp.Service; package org.opentcs.kc.udp.Service;
import java.util.HashMap;
import org.opentcs.drivers.vehicle.MovementCommand;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
import org.opentcs.kc.udp.agv.param.AgvEvent; import org.opentcs.kc.udp.agv.param.AgvEvent;
import org.opentcs.kc.udp.agv.param.AgvEventConstant; import org.opentcs.kc.udp.agv.param.AgvEventConstant;
@ -11,7 +13,9 @@ import org.opentcs.kc.udp.agv.param.function.navigation.Point;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
public class HybridNavigation extends BaseCommand{ public class HybridNavigation
extends
BaseCommand {
/** /**
* 测试模式true=开启测试false=关闭测试 * 测试模式true=开启测试false=关闭测试
@ -38,6 +42,14 @@ public class HybridNavigation extends BaseCommand{
* 用于定位段在整个任务中的位置目的是区分同一个段ID是否在一个任务中出现多次从1开始奇数递增例如1->3->5->7 * 用于定位段在整个任务中的位置目的是区分同一个段ID是否在一个任务中出现多次从1开始奇数递增例如1->3->5->7
*/ */
private static Integer pathSerialNum = 1; private static Integer pathSerialNum = 1;
/**
* 订单名映射int类型数据.
*/
private static HashMap<String, Integer> orderNameMap = new HashMap<>();
/**
* 订单映射最大订单ID.
*/
private static int currentMaxiOrderId = 0;
/** /**
@ -45,45 +57,37 @@ public class HybridNavigation extends BaseCommand{
* 指令0xAE * 指令0xAE
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent navigationControl(Integer sourcePoint, Integer destinationPoint, Integer pathID, String operation) { public static AgvEvent navigationControl(
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_STATUS); Integer sourcePoint, Integer destinationPoint, Integer pathID, String operation
) {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_MIXED_ISSUANCE_TASK);
//TODO 构建 //TODO 构建
Integer orderId = orderID;
// Integer orderId = 1;
//构建point
Action[] pointActions1 = new Action[]{
new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01))
//,new Action()... 每一个point 可以绑定一个或者多个 action
};
Action[] pointActions2 = new Action[]{
new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01))
};
// Action[] pointActions3 = new Action[]{
// new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01))
// };
Integer oldPointSerialNum = pointSerialNum; Integer oldPointSerialNum = pointSerialNum;
pointSerialNum += 2; pointSerialNum += 2;
Point[] points = new Point[]{ Point[] points = new Point[]{
new Point(oldPointSerialNum, sourcePoint, 1f, (byte)0x00, ByteUtils.usintTo1Byte(pointActions1.length),pointActions1), new Point(
new Point(pointSerialNum, destinationPoint, 1f, (byte)0x00, ByteUtils.usintTo1Byte(pointActions2.length),pointActions2) oldPointSerialNum, sourcePoint, 0f, (byte) 0x00, ByteUtils.usintTo1Byte(0), null
// new Point(4, 3, 1f, (byte)0x00, ByteUtils.usintTo1Byte(pointActions3.length),pointActions3) ),
new Point(
pointSerialNum, destinationPoint, 0f, (byte) 0x00, ByteUtils.usintTo1Byte(0), null
)
}; };
//构建path
Action[] pathActions1 = new Action[]{
new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01))
//,new Action()... 每一个path 可以绑定一个或者多个 action
};
// Action[] pathActions2 = new Action[]{
// new Action(ActionSet.stop0x01, (byte) 0x00, 1, ActionSet.stop0x01_paramsize, ActionSet.stop0x01(orderId, (byte) 0x01))
// };
Path[] paths = new Path[]{ Path[] paths = new Path[]{
new Path(pathSerialNum,pathID,1f,(byte)0x00,(byte)0x00,ByteUtils.usintTo1Byte(pathActions1.length),5f,1f,pathActions1) , new Path(
// new Path(3,2,1f,(byte)0x00,(byte)0x00,ByteUtils.usintTo1Byte(pathActions2.length),5f,1f,pathActions2) , pathSerialNum, pathID, 0f, (byte) 0x00, (byte) 0x01, ByteUtils.usintTo1Byte(0), 1f, 1f, null
),
}; };
NavigationParam navigationParam = new NavigationParam(orderID,taskKey,ByteUtils.usintTo1Byte(points.length),ByteUtils.usintTo1Byte(points.length-1),points,paths); pathSerialNum += 2;
NavigationParam navigationParam = new NavigationParam(
orderID, taskKey, ByteUtils.usintTo1Byte(points.length), ByteUtils.usintTo1Byte(
points.length - 1
), points, paths
);
taskKey++;
agvEvent.setBody(navigationParam.toBytes()); agvEvent.setBody(navigationParam.toBytes());
return agvEvent; return agvEvent;
@ -96,16 +100,24 @@ public class HybridNavigation extends BaseCommand{
* @param destinationPoint 下发终点 * @param destinationPoint 下发终点
* @param operation 执行操作 * @param operation 执行操作
*/ */
public static void command(String orderName, String sourcePoint, String destinationPoint, String operation) { public static void command(
if (TEST_MODEL) { String orderName, String sourcePoint, String destinationPoint, String operation
return; ) {
} // if (TEST_MODEL) {
// return;
// }
Integer newOrderName = Integer.parseInt(orderName); Integer newOrderName = getUniqueOrderID(orderName);
Integer newSourcePoint = Integer.parseInt(sourcePoint); Integer newSourcePoint = Integer.parseInt(sourcePoint);
Integer newDestinationPoint = Integer.parseInt(destinationPoint); Integer newDestinationPoint = Integer.parseInt(destinationPoint);
//拼接起点终点字符串转为Integer类型获取唯一pathID //拼接起点终点字符串转为Integer类型获取唯一pathID
Integer pathID = Integer.parseInt(sourcePoint + destinationPoint); Integer pathID;
if (newSourcePoint <= newDestinationPoint) {
pathID = Integer.parseInt(sourcePoint + destinationPoint);
} else {
pathID = Integer.parseInt(destinationPoint + sourcePoint);
}
System.out.println("pathID:" + pathID);
if (!orderID.equals(newOrderName)) { if (!orderID.equals(newOrderName)) {
//切换订单重置参数 //切换订单重置参数
@ -116,17 +128,19 @@ public class HybridNavigation extends BaseCommand{
AgvEvent agvEvent = navigationControl(newSourcePoint, newDestinationPoint, pathID, operation); AgvEvent agvEvent = navigationControl(newSourcePoint, newDestinationPoint, pathID, operation);
printInfo(agvEvent); printInfo(agvEvent);
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ if (rcv.isOk()) {
System.out.println("0xAE ok"); System.out.println("0xAE ok");
}else { }
else {
System.out.println(); System.out.println();
System.out.println("0xAE fail"); System.out.println("0xAE fail");
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
} }
} }
/** /**
* 初始化参数 * 初始化参数
*
* @param newOrderName 新的订单ID * @param newOrderName 新的订单ID
*/ */
private static void initParams(Integer newOrderName) { private static void initParams(Integer newOrderName) {
@ -136,4 +150,34 @@ public class HybridNavigation extends BaseCommand{
pathSerialNum = 1; pathSerialNum = 1;
} }
/**
* 维护订单名对应int类型唯一ID
* @param orderName 订单名
* @return Integer
*/
private static Integer getUniqueOrderID(String orderName){
Integer orderId;
if (orderNameMap.containsKey(orderName)) {
//订单名已存在
orderId = orderNameMap.get(orderName);
} else {
//订单名不存在创建对应映射
currentMaxiOrderId = currentMaxiOrderId + 1;
orderId = currentMaxiOrderId;
orderNameMap.put(orderName, orderId);
}
return orderId;
}
/**
* 任务结束删除唯一orderID.
* @param command 订单
*/
public static void delUniqueOrderID(MovementCommand command){
String orderName = command.getTransportOrder().getName();
orderNameMap.remove(orderName);
}
} }

View File

@ -7,12 +7,9 @@ import org.opentcs.kc.udp.agv.param.function.x14.RobotSetPosition;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
/** public class ManualPosition
* @author xzh extends
* @date 2025/2/21 BaseCommand {
* @desc 手动定位 0x14
*/
public class ManualPosition extends BaseCommand{
public static AgvEvent manualLocation(double agvX, double agvY, double agvAngle) { public static AgvEvent manualLocation(double agvX, double agvY, double agvAngle) {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_ROBOT_SET_POSITION); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_ROBOT_SET_POSITION);
@ -27,19 +24,21 @@ public class ManualPosition extends BaseCommand{
AgvEvent agvEvent = manualLocation(agvX, agvY, agvAngle); AgvEvent agvEvent = manualLocation(agvX, agvY, agvAngle);
printInfo(agvEvent); printInfo(agvEvent);
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ if (rcv.isOk()) {
System.out.println(); System.out.println();
System.out.println("received "+ "isok:"+rcv.isOk()+" dataBytes:"); System.out.println("received " + "isok:" + rcv.isOk() + " dataBytes:");
printInfo(rcv); printInfo(rcv);
if(rcv.isOk()){ if (rcv.isOk()) {
//get and parse value //get and parse value
System.out.println("0x14 ok"); System.out.println("0x14 ok");
}else { }
else {
System.out.println("0x14 failed"); System.out.println("0x14 failed");
} }
}else { }
else {
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
throw new RuntimeException("0x14 failed"); throw new RuntimeException("0x14 failed");
} }
} }

View File

@ -6,19 +6,16 @@ import org.opentcs.kc.udp.agv.param.function.x17.QueryRobotRunStatusRsp;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
/** public class QryRobotRunStatus
* @author xzh extends
* @date 2025/2/21 BaseCommand {
* @desc 获取AGV运行状态
*/
public class QryRobotRunStatus extends BaseCommand{
/** /**
* decs: 查询机器人运行状态 * decs: 查询机器人运行状态.
* 指令0x17 * 指令0x17
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent queryRobotRunStatus() { public static AgvEvent queryRobotRunStatus() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_RUN_STATUS); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_RUN_STATUS);
return agvEvent; return agvEvent;
@ -29,19 +26,22 @@ public class QryRobotRunStatus extends BaseCommand{
AgvEvent agvEvent = queryRobotRunStatus(); AgvEvent agvEvent = queryRobotRunStatus();
printInfo(agvEvent); printInfo(agvEvent);
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ if (rcv.isOk()) {
//QueryCargoStatusRsp queryCargoStatusRsp = new QueryCargoStatusRsp(rcv.getDataBytes()); //QueryCargoStatusRsp queryCargoStatusRsp = new QueryCargoStatusRsp(rcv.getDataBytes());
QueryRobotRunStatusRsp queryRobotRunStatusRsp = new QueryRobotRunStatusRsp(rcv.getDataBytes()); QueryRobotRunStatusRsp queryRobotRunStatusRsp = new QueryRobotRunStatusRsp(
rcv.getDataBytes()
);
System.out.println(queryRobotRunStatusRsp.toString()); System.out.println(queryRobotRunStatusRsp.toString());
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
for (byte b:rcv.getValue()){ for (byte b : rcv.getValue()) {
System.out.print(byteToHex(b)+" "); System.out.print(byteToHex(b) + " ");
} }
return queryRobotRunStatusRsp; return queryRobotRunStatusRsp;
}else { }
else {
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
throw new RuntimeException("0x17 fail"); throw new RuntimeException("0x17 fail");
} }
} }

View File

@ -6,20 +6,17 @@ import org.opentcs.kc.udp.agv.param.function.af.QueryRobotStatusRsp;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
/**
* @author xzh
* @date 2025/2/21
* @desc 查询机器人状态
*/
public class QryRobotStatus public class QryRobotStatus
extends BaseCommand { extends
BaseCommand {
/** /**
* decs: 查询机器人状态 * decs: 查询机器人状态
* 指令0xAF * 指令0xAF
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent queryStatus() { public static AgvEvent queryStatus() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_STATUS); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_QUERY_ROBOT_STATUS);
return agvEvent; return agvEvent;
@ -31,16 +28,22 @@ public class QryRobotStatus
printInfo(agvEvent); printInfo(agvEvent);
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ if(rcv.isOk()){
QueryRobotStatusRsp queryRobotStatusRsp = new QueryRobotStatusRsp(rcv.getDataBytes());
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); for (byte b:rcv.getValue()){
return queryRobotStatusRsp; System.out.print(byteToHex(b)+" ");
// for (byte b:rcv.getValue()){ }
// System.out.print(byteToHex(b)+" "); System.out.println();
System.out.println("0xAF received transationId : "+ "isok:"+rcv.isOk());
// for (byte c:rcv.getDataBytes()){
// System.out.print(byteToHex(c)+" ");
// } // }
}else { QueryRobotStatusRsp queryRobotStatusRsp = new QueryRobotStatusRsp(rcv.getDataBytes());
return queryRobotStatusRsp;
}
else {
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
throw new RuntimeException("0xAF fail"); throw new RuntimeException("0xAF fail");
} }
} }

View File

@ -13,60 +13,64 @@ import org.opentcs.kc.udp.agv.param.function.b1.SubscribeRsp;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
/** public class SubCargoStatus
* @author xzh extends
* @date 2025/2/21 BaseCommand {
* @desc 订阅载货状态
*/ public static final int intervalTime = 1000;
public class SubCargoStatus extends BaseCommand{
/** /**
* decs: 下发订阅信息 * decs: 下发订阅信息
* 指令0xB1 * 指令0xB1
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent issueSubscribe() { public static AgvEvent issueSubscribe() {
List<SubscribeInfo> subscribeInfoList = new ArrayList<>(); List<SubscribeInfo> subscribeInfoList = new ArrayList<>();
// SubscribeInfo subscribeInfo = new SubscribeInfo(new byte[]{(byte)0xaf,(byte)0x00}, (short) 100,1000); // SubscribeInfo subscribeInfo = new SubscribeInfo(new byte[]{(byte)0xaf,(byte)0x00}, (short) 100,1000);
SubscribeInfo subscribeInfo = new SubscribeInfo(new byte[]{(byte)0xb0,(byte)0x00}, (short) 100,1000); SubscribeInfo subscribeInfo = new SubscribeInfo(
new byte[]{(byte) 0xb0, (byte) 0x00}, (short) intervalTime, 10000
);
subscribeInfoList.add(subscribeInfo); subscribeInfoList.add(subscribeInfo);
SubscribeParam subscribeParam = new SubscribeParam(subscribeInfoList); SubscribeParam subscribeParam = new SubscribeParam(subscribeInfoList);
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_ISSUE_SUBSCRIPTION,subscribeParam.toBytes()); AgvEvent agvEvent = new AgvEvent(
AgvEventConstant.CommandCode_ISSUE_SUBSCRIPTION, subscribeParam.toBytes()
);
return agvEvent; return agvEvent;
} }
public static void command() public static void command() {
{
// 0xB1(订阅信息) // 0xB1(订阅信息)
AgvEvent agvEvent = issueSubscribe(); AgvEvent agvEvent = issueSubscribe();
printInfo(agvEvent); printInfo(agvEvent);
//todo 订阅参数构建完毕 去写 回调部分 //todo 订阅参数构建完毕 去写 回调部分
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ if (rcv.isOk()) {
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
SubscribeRsp subscribeRsp = new SubscribeRsp(rcv.getDataBytes()); SubscribeRsp subscribeRsp = new SubscribeRsp(rcv.getDataBytes());
if(subscribeRsp.isOk()){ if (subscribeRsp.isOk()) {
//...
}else {
//... //...
} }
}else { else {
//...
}
}
else {
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
} }
} }
//订阅执行操作 //订阅执行操作
public void subscribe0xB0Operate(String name,QueryCargoStatusRsp queryCargoStatusRsp) public void subscribe0xB0Operate(String name, QueryCargoStatusRsp queryCargoStatusRsp) {
{
// Vehicle vehicle = vehicleService.fetchObject(Vehicle.class, name); // Vehicle vehicle = vehicleService.fetchObject(Vehicle.class, name);
//修改载货状态 //修改载货状态
if (queryCargoStatusRsp.isCargo == 0) { if (queryCargoStatusRsp.isCargo == 0) {
//未载货 //未载货
} else if (queryCargoStatusRsp.isCargo == 1) { }
else if (queryCargoStatusRsp.isCargo == 1) {
//载货 //载货
} }
} }

View File

@ -10,27 +10,29 @@ import org.opentcs.kc.udp.agv.param.function.b1.SubscribeRsp;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
/** public class SubRobotStatue
* @author xzh extends
* @date 2025/2/21 BaseCommand {
* @desc 订阅机器人状态
*/
public class SubRobotStatue extends BaseCommand {
public static final int intervalTime = 1000;
/** /**
* decs: 下发订阅信息 * decs: 下发订阅信息
* 指令0xB1 * 指令0xB1
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent issueSubscribe() { public static AgvEvent issueSubscribe() {
List<SubscribeInfo> subscribeInfoList = new ArrayList<>(); List<SubscribeInfo> subscribeInfoList = new ArrayList<>();
SubscribeInfo subscribeInfo = new SubscribeInfo(new byte[]{(byte)0xaf,(byte)0x00}, (short) 100,1000); SubscribeInfo subscribeInfo = new SubscribeInfo(
new byte[]{(byte) 0xaf, (byte) 0x00}, (short) intervalTime, 10000
);
// SubscribeInfo subscribeInfo = new SubscribeInfo(new byte[]{(byte)0xb0,(byte)0x00}, (short) 100,1000); // SubscribeInfo subscribeInfo = new SubscribeInfo(new byte[]{(byte)0xb0,(byte)0x00}, (short) 100,1000);
subscribeInfoList.add(subscribeInfo); subscribeInfoList.add(subscribeInfo);
SubscribeParam subscribeParam = new SubscribeParam(subscribeInfoList); SubscribeParam subscribeParam = new SubscribeParam(subscribeInfoList);
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_ISSUE_SUBSCRIPTION,subscribeParam.toBytes()); AgvEvent agvEvent = new AgvEvent(
AgvEventConstant.CommandCode_ISSUE_SUBSCRIPTION, subscribeParam.toBytes()
);
return agvEvent; return agvEvent;
} }
@ -40,18 +42,20 @@ public class SubRobotStatue extends BaseCommand {
printInfo(agvEvent); printInfo(agvEvent);
//todo 订阅参数构建完毕 去写 回调部分 //todo 订阅参数构建完毕 去写 回调部分
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ if (rcv.isOk()) {
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
SubscribeRsp subscribeRsp = new SubscribeRsp(rcv.getDataBytes()); SubscribeRsp subscribeRsp = new SubscribeRsp(rcv.getDataBytes());
if(subscribeRsp.isOk()){ if (subscribeRsp.isOk()) {
//...
}else {
//... //...
} }
}else { else {
//...
}
}
else {
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
} }
} }
} }

View File

@ -12,31 +12,30 @@ import org.opentcs.kc.udp.agv.param.function.write.WriteValueMember;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
/** public class SwitchAutomaticMode
* @author xzh extends
* @date 2025/2/21 BaseCommand {
* @desc 切换自动模式
*/
public class SwitchAutomaticMode extends BaseCommand{
/** /**
* decs: write操作 * decs: write操作
* 指令0x03 * 指令0x03
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent writeValue() { public static AgvEvent writeValue() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_WRITE); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_WRITE);
List<WriteValueMember> valueMemberList = new ArrayList<>(); List<WriteValueMember> valueMemberList = new ArrayList<>();
WriteValueMember valueMember1 = new WriteValueMember(Short.valueOf("1"),Short.valueOf("4"), ByteUtils.uintToBytes(3, ByteOrder.LITTLE_ENDIAN)); WriteValueMember valueMember1 = new WriteValueMember(
Short.valueOf("1"), Short.valueOf("4"), ByteUtils.uintToBytes(3, ByteOrder.LITTLE_ENDIAN)
);
valueMemberList.add(valueMember1); valueMemberList.add(valueMember1);
List<WriteStrValue> strValueList = new ArrayList<>(); List<WriteStrValue> strValueList = new ArrayList<>();
WriteStrValue strValue = new WriteStrValue("TestRW", 1, valueMemberList); WriteStrValue strValue = new WriteStrValue("TestRW", 1, valueMemberList);
strValueList.add(strValue); strValueList.add(strValue);
WriteParam param = new WriteParam(1,strValueList ); WriteParam param = new WriteParam(1, strValueList);
agvEvent.setBody(param.toBytes()); agvEvent.setBody(param.toBytes());
return agvEvent; return agvEvent;
@ -47,19 +46,21 @@ public class SwitchAutomaticMode extends BaseCommand{
AgvEvent agvEvent = writeValue(); AgvEvent agvEvent = writeValue();
printInfo(agvEvent); printInfo(agvEvent);
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ if (rcv.isOk()) {
System.out.println(); System.out.println();
System.out.println("received "+ "isok:"+rcv.isOk()+" dataBytes:"); System.out.println("received " + "isok:" + rcv.isOk() + " dataBytes:");
printInfo(rcv); printInfo(rcv);
if(rcv.isOk()){ if (rcv.isOk()) {
//get and parse value //get and parse value
System.out.println("write ok"); System.out.println("write ok");
}else { }
else {
System.out.println("write failed"); System.out.println("write failed");
} }
}else { }
else {
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
} }
} }
} }

View File

@ -12,31 +12,30 @@ import org.opentcs.kc.udp.agv.param.function.write.WriteValueMember;
import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage; import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
/** public class SwitchManualMode
* @author xzh extends
* @date 2025/2/21 BaseCommand {
* @desc 切换手动模式
*/
public class SwitchManualMode extends BaseCommand{
/** /**
* decs: write操作 * decs: write操作
* 指令0x03 * 指令0x03
* author: caixiang * author: caixiang
* date: 2025/1/17 16:25 * date: 2025/1/17 16:25
* */ */
public static AgvEvent writeValue() { public static AgvEvent writeValue() {
AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_WRITE); AgvEvent agvEvent = new AgvEvent(AgvEventConstant.CommandCode_WRITE);
List<WriteValueMember> valueMemberList = new ArrayList<>(); List<WriteValueMember> valueMemberList = new ArrayList<>();
WriteValueMember valueMember1 = new WriteValueMember(Short.valueOf("0"),Short.valueOf("4"), ByteUtils.uintToBytes(3, ByteOrder.LITTLE_ENDIAN)); WriteValueMember valueMember1 = new WriteValueMember(
Short.valueOf("0"), Short.valueOf("4"), ByteUtils.uintToBytes(3, ByteOrder.LITTLE_ENDIAN)
);
valueMemberList.add(valueMember1); valueMemberList.add(valueMember1);
List<WriteStrValue> strValueList = new ArrayList<>(); List<WriteStrValue> strValueList = new ArrayList<>();
WriteStrValue strValue = new WriteStrValue("TestRW", 1, valueMemberList); WriteStrValue strValue = new WriteStrValue("TestRW", 1, valueMemberList);
strValueList.add(strValue); strValueList.add(strValue);
WriteParam param = new WriteParam(1,strValueList ); WriteParam param = new WriteParam(1, strValueList);
agvEvent.setBody(param.toBytes()); agvEvent.setBody(param.toBytes());
return agvEvent; return agvEvent;
@ -47,19 +46,21 @@ public class SwitchManualMode extends BaseCommand{
AgvEvent agvEvent = writeValue(); AgvEvent agvEvent = writeValue();
printInfo(agvEvent); printInfo(agvEvent);
RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent); RcvEventPackage rcv = UDPClient.localAGV.send(agvEvent);
if(rcv.isOk()){ if (rcv.isOk()) {
System.out.println(); System.out.println();
System.out.println("received "+ "isok:"+rcv.isOk()+" dataBytes:"); System.out.println("received " + "isok:" + rcv.isOk() + " dataBytes:");
printInfo(rcv); printInfo(rcv);
if(rcv.isOk()){ if (rcv.isOk()) {
//get and parse value //get and parse value
System.out.println("write ok"); System.out.println("write ok");
}else { }
else {
System.out.println("write failed"); System.out.println("write failed");
} }
}else { }
else {
System.out.println(); System.out.println();
System.out.println("received transationId : "+ "isok:"+rcv.isOk()); System.out.println("received transationId : " + "isok:" + rcv.isOk());
} }
} }
} }

View File

@ -7,7 +7,9 @@ import org.opentcs.kc.udp.io.UDPClient;
/** /**
* 作者蔡翔 * 作者蔡翔
*/ */
public class AgvUdpChannelInitializer extends ChannelInitializer<NioDatagramChannel> { public class AgvUdpChannelInitializer
extends
ChannelInitializer<NioDatagramChannel> {
private UDPClient client; private UDPClient client;
public AgvUdpChannelInitializer(UDPClient client) { public AgvUdpChannelInitializer(UDPClient client) {
@ -17,7 +19,8 @@ public class AgvUdpChannelInitializer extends ChannelInitializer<NioDatagramChan
@Override @Override
protected void initChannel(NioDatagramChannel channel) throws Exception { protected void initChannel(NioDatagramChannel channel)
throws Exception {
// Modbus // Modbus
// 在管道中添加我们自己的接收数据实现方法 // 在管道中添加我们自己的接收数据实现方法
//todo 到时候这里改成 FixedLengthFrameDecoder 资料https://www.cnblogs.com/java-chen-hao/p/11571229.html //todo 到时候这里改成 FixedLengthFrameDecoder 资料https://www.cnblogs.com/java-chen-hao/p/11571229.html

View File

@ -11,12 +11,9 @@ import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.opentcs.kc.udp.io.UDPClient; import org.opentcs.kc.udp.io.UDPClient;
/** public class AgvUdpDecode
* @Desc: "" extends
* @Author: caixiang SimpleChannelInboundHandler<DatagramPacket> {
* @DATE: 2024-06-06 10:45
*/
public class AgvUdpDecode extends SimpleChannelInboundHandler<DatagramPacket> {
private UDPClient client; private UDPClient client;
@ -32,13 +29,15 @@ public class AgvUdpDecode extends SimpleChannelInboundHandler<DatagramPacket> {
throws Exception { throws Exception {
//获得应答DatagramPacket提供了content()方法取得报文的实际内容 //获得应答DatagramPacket提供了content()方法取得报文的实际内容
ByteBuf in = msg.content(); ByteBuf in = msg.content();
if (in.readableBytes() < HEADER_SIZE){ if (in.readableBytes() < HEADER_SIZE) {
throw new Exception("readed bytes < header length"); throw new Exception("readed bytes < header length");
} }
int dataLength = in.getShortLE(24); int dataLength = in.getShortLE(24);
if (dataLength < 0) { if (dataLength < 0) {
throw new Exception("bodyLength [" + dataLength + "] is not right, remote:" + ctx.channel().remoteAddress()); throw new Exception(
"bodyLength [" + dataLength + "] is not right, remote:" + ctx.channel().remoteAddress()
);
} }
int neededLength = HEADER_SIZE + dataLength; int neededLength = HEADER_SIZE + dataLength;
@ -46,35 +45,35 @@ public class AgvUdpDecode extends SimpleChannelInboundHandler<DatagramPacket> {
//收到的数据是否足够组包 //收到的数据是否足够组包
if(isDataEnough<0){ if (isDataEnough < 0) {
// 不够消息体长度(剩下的buffe组不了消息体),重新去组包 // 不够消息体长度(剩下的buffe组不了消息体),重新去组包
throw new Exception("readed bytes < content length"); throw new Exception("readed bytes < content length");
}else { }
else {
//todo这里重写subscribe 的逻辑,注意要区分是 订阅的还是 主动请求的 //todo这里重写subscribe 的逻辑,注意要区分是 订阅的还是 主动请求的
//组包成功 //组包成功
byte[] body = new byte[neededLength]; byte[] body = new byte[neededLength];
in.readBytes(body); in.readBytes(body);
//System.out.println("received bytes :"+ Arrays.toString(body)); //System.out.println("received bytes :"+ Arrays.toString(body));
String uuid = body[18]+"-"+body[19]; String uuid = body[18] + "-" + body[19];
Package mbPackage = new Package(body,uuid); Package mbPackage = new Package(body, uuid);
byte commandCode = body[21]; byte commandCode = body[21];
if(body[18]==(byte)0x00 && body[19]==(byte)0x00){ if (body[18] == (byte) 0x00 && body[19] == (byte) 0x00) {
//获取响应IP //获取响应IP
InetSocketAddress sender = msg.sender(); InetSocketAddress sender = msg.sender();
String hostAddress = sender.getAddress().getHostAddress(); String hostAddress = sender.getAddress().getHostAddress();
if(commandCode == (byte)0xAF ){ if (commandCode == (byte) 0xAF || commandCode == (byte) 0xB0) {
client.subscribe0xAF(new RcvEventPackage(body[22],body)); client.subscribeKC(new RcvEventPackage(body[22], body), body);
}else if(commandCode == (byte)0xB0){
client.subscribe0xB0(new RcvEventPackage(body[22],body));
}else {
SendedList.set(uuid , mbPackage);
} }
}else { else {
SendedList.set(uuid , mbPackage); SendedList.set(uuid, mbPackage);
}
}
else {
SendedList.set(uuid, mbPackage);
} }
} }
} }

View File

@ -5,15 +5,15 @@ import java.util.Arrays;
import java.util.List; import java.util.List;
import org.opentcs.kc.common.Package; import org.opentcs.kc.common.Package;
import org.opentcs.kc.common.byteutils.ByteUtil; import org.opentcs.kc.common.byteutils.ByteUtil;
//import org.opentcs.kc.common.byteutils.ByteUtil; // import org.opentcs.kc.common.byteutils.ByteUtil;
//import org.opentcs.kc.common.byteutils.ByteUtil; // import org.opentcs.kc.common.byteutils.ByteUtil;
/** public class AgvEvent
* @Desc: "" extends
* @Author: caixiang AgvEventHeader
* @DATE: 2024/12/27 15:59 implements
*/ IAgvEvent,
public class AgvEvent extends AgvEventHeader implements IAgvEvent, Serializable { Serializable {
private byte[] bodyLength; private byte[] bodyLength;
private byte[] retain; private byte[] retain;
private byte[] body; private byte[] body;
@ -23,16 +23,17 @@ public class AgvEvent extends AgvEventHeader implements IAgvEvent, Serializable
public AgvEvent(Byte commandCode) { public AgvEvent(Byte commandCode) {
super(commandCode); super(commandCode);
//初始化 //初始化
bodyLength = new byte[]{0x00,0x00}; bodyLength = new byte[]{0x00, 0x00};
retain = new byte[]{0x00,0x00}; retain = new byte[]{0x00, 0x00};
body = new byte[]{}; body = new byte[]{};
} }
public AgvEvent(Byte commandCode,byte[] body) {
public AgvEvent(Byte commandCode, byte[] body) {
super(commandCode); super(commandCode);
//初始化 //初始化
bodyLength = new byte[]{0x00,0x00}; bodyLength = new byte[]{0x00, 0x00};
retain = new byte[]{0x00,0x00}; retain = new byte[]{0x00, 0x00};
if(commandCode.equals(AgvEventConstant.CommandCode_ISSUE_SUBSCRIPTION)){ if (commandCode.equals(AgvEventConstant.CommandCode_ISSUE_SUBSCRIPTION)) {
//依据命令码 构建参数 //依据命令码 构建参数
this.bodyLength = ByteUtil.shortToBytes((short) body.length); this.bodyLength = ByteUtil.shortToBytes((short) body.length);
this.body = body; this.body = body;
@ -69,7 +70,7 @@ public class AgvEvent extends AgvEventHeader implements IAgvEvent, Serializable
} }
public String getTransationIdString() { public String getTransationIdString() {
return transationId[0]+"-"+transationId[1]; return transationId[0] + "-" + transationId[1];
} }
@Override @Override

View File

@ -1,13 +1,10 @@
package org.opentcs.kc.udp.agv.param; package org.opentcs.kc.udp.agv.param;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2024/12/27 10:19
*/
public class AgvEventConstant { public class AgvEventConstant {
public static final byte[] AuthorizationCode = new byte[]{ (byte)0xd4 , (byte)0x97 , (byte)0x44 , (byte)0x9c , (byte)0xcb , (byte)0xcf , (byte)0x0b , (byte)0x4c , (byte)0x95 , (byte)0x51 , (byte)0xd8 , (byte)0x61 , (byte)0x70 , (byte)0xf1 , (byte)0xe7 , (byte)0x94}; public static final byte[] AuthorizationCode = new byte[]{(byte) 0xd4, (byte) 0x97, (byte) 0x44,
(byte) 0x9c, (byte) 0xcb, (byte) 0xcf, (byte) 0x0b, (byte) 0x4c, (byte) 0x95, (byte) 0x51,
(byte) 0xd8, (byte) 0x61, (byte) 0x70, (byte) 0xf1, (byte) 0xe7, (byte) 0x94};
public static final byte VersionNum = 0x01; public static final byte VersionNum = 0x01;
@ -25,13 +22,13 @@ public class AgvEventConstant {
public static final byte CommandCode_QUERY_CARRY_STATUS = (byte) 0xB0; public static final byte CommandCode_QUERY_CARRY_STATUS = (byte) 0xB0;
public static final byte CommandCode_ISSUE_SUBSCRIPTION = (byte) 0xB1; public static final byte CommandCode_ISSUE_SUBSCRIPTION = (byte) 0xB1;
public static final byte CommandCode_ACT_IMMEDIATELY = (byte) 0xB2; public static final byte CommandCode_ACT_IMMEDIATELY = (byte) 0xB2;
public static final byte CommandCode_SET_ABILITY= (byte) 0xB7; public static final byte CommandCode_SET_ABILITY = (byte) 0xB7;
public static final byte CommandCode_ROBOT_SET_POSITION= (byte) 0x14; public static final byte CommandCode_ROBOT_SET_POSITION = (byte) 0x14;
public static final byte CommandCode_GET_ROBOT_POSITION= (byte) 0x15; public static final byte CommandCode_GET_ROBOT_POSITION = (byte) 0x15;
public static final byte CommandCode_NAVIGATION_CONTROL= (byte) 0x16; public static final byte CommandCode_NAVIGATION_CONTROL = (byte) 0x16;
public static final byte CommandCode_QUERY_ROBOT_NAVIGATION_STATUS= (byte) 0x1D; public static final byte CommandCode_QUERY_ROBOT_NAVIGATION_STATUS = (byte) 0x1D;
public static final byte CommandCode_QUERY_ROBOT_RUN_STATUS= (byte) 0x17; public static final byte CommandCode_QUERY_ROBOT_RUN_STATUS = (byte) 0x17;
public static final byte CommandCode_CONFIRM_ROBOT_POSITION= (byte) 0x1F; public static final byte CommandCode_CONFIRM_ROBOT_POSITION = (byte) 0x1F;
//命令码 结束 //命令码 结束

View File

@ -6,11 +6,6 @@ import java.util.List;
import org.opentcs.kc.common.CaffeineUtil; import org.opentcs.kc.common.CaffeineUtil;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2024-06-07 13:29
*/
public class AgvEventHeader { public class AgvEventHeader {
@ -50,11 +45,11 @@ public class AgvEventHeader {
this.retain = 0x00; this.retain = 0x00;
} }
public AgvEventHeader(byte[] src){ public AgvEventHeader(byte[] src) {
this.authorizationCode = ByteUtils.copyOfRange(src,0,16); this.authorizationCode = ByteUtils.copyOfRange(src, 0, 16);
this.versionNum = src[16]; this.versionNum = src[16];
this.msgType = src[17]; this.msgType = src[17];
this.transationId = ByteUtils.copyOfRange(src,18,20); this.transationId = ByteUtils.copyOfRange(src, 18, 20);
this.serviceCode = src[20]; this.serviceCode = src[20];
this.commandCode = src[21]; this.commandCode = src[21];
this.executionCode = src[22]; this.executionCode = src[22];
@ -88,7 +83,7 @@ public class AgvEventHeader {
'}'; '}';
} }
public List<Byte> getHeaderBytes(){ public List<Byte> getHeaderBytes() {
List<Byte> bytes = new ArrayList<>(); List<Byte> bytes = new ArrayList<>();
//add 授权码 //add 授权码
for (byte b : authorizationCode) { for (byte b : authorizationCode) {

View File

@ -1,10 +1,12 @@
package org.opentcs.kc.udp.agv.param; package org.opentcs.kc.udp.agv.param;
import org.opentcs.kc.common.byteutils.ByteUtil;
import org.opentcs.kc.common.Package; import org.opentcs.kc.common.Package;
import org.opentcs.kc.common.byteutils.ByteUtil;
public interface IAgvEvent { public interface IAgvEvent {
//read sended //read sended
public Package toBytes(); public Package toBytes();
//write sended //write sended
public Package toBytes(Object newValue); public Package toBytes(Object newValue);

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.af;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/21 10:03
*/
public class AbnormalEventStatusInfo { public class AbnormalEventStatusInfo {
private byte[] src; private byte[] src;
//事件码,2个字节 //事件码,2个字节

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.af;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/21 10:38
*/
public class ActionInfo { public class ActionInfo {
private byte[] src; private byte[] src;
//动作 ID,4个字节 //动作 ID,4个字节
@ -20,6 +15,6 @@ public class ActionInfo {
this.src = src; this.src = src;
this.actionId = ByteUtils.copyBytes(src, 0, 4); this.actionId = ByteUtils.copyBytes(src, 0, 4);
this.actionStatus = src[4]; this.actionStatus = src[4];
this.remain = ByteUtils.copyBytes(src,5,7); this.remain = ByteUtils.copyBytes(src, 5, 7);
} }
} }

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.af;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/20 16:28
*/
public class BatteryStatusInfo { public class BatteryStatusInfo {
//src //src
private byte[] src; private byte[] src;
@ -20,15 +15,16 @@ public class BatteryStatusInfo {
public byte chargingState; public byte chargingState;
//预留,7个字节 //预留,7个字节
public byte[] remain; public byte[] remain;
public BatteryStatusInfo(byte[] src) { public BatteryStatusInfo(byte[] src) {
this.src = src; this.src = src;
this.batteryPercentage = ByteUtils.bytesToFloat(src, 0); this.batteryPercentage = ByteUtils.bytesToFloat(src, 0);
this.voltage = ByteUtils.bytesToFloat(src, 4); this.voltage = ByteUtils.bytesToFloat(src, 4);
this.electricCurrent = ByteUtils.bytesToFloat(src, 8); this.electricCurrent = ByteUtils.bytesToFloat(src, 8);
this.chargingState = src[12]; this.chargingState = src[12];
this.remain=new byte[7]; this.remain = new byte[7];
for (int i = 0; i < 7; i++) { for (int i = 0; i < 7; i++) {
this.remain[i]=src[13+i]; this.remain[i] = src[13 + i];
} }
} }

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.af;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/20 14:56
*/
public class LocationStatusInfo { public class LocationStatusInfo {
private byte[] src; private byte[] src;
//4个字节, 全局x,单位 m //4个字节, 全局x,单位 m
@ -42,15 +37,15 @@ public class LocationStatusInfo {
src[9] = (byte) 0x3c; src[9] = (byte) 0x3c;
src[10] = (byte) 0x1d; src[10] = (byte) 0x1d;
src[11] = (byte) 0x3d; src[11] = (byte) 0x3d;
System.out.println("x: "+ByteUtils.bytesToFloat(src, 0)); System.out.println("x: " + ByteUtils.bytesToFloat(src, 0));
System.out.println("y: "+ByteUtils.bytesToFloat(src, 4)); System.out.println("y: " + ByteUtils.bytesToFloat(src, 4));
System.out.println("z: "+ByteUtils.bytesToFloat(src, 8)); System.out.println("z: " + ByteUtils.bytesToFloat(src, 8));
} }
public LocationStatusInfo(byte[] src) { public LocationStatusInfo(byte[] src) {
this.src = src; this.src = src;
for (byte b:src){ for (byte b : src) {
System.out.print(byteToHex(b)+" "); System.out.print(byteToHex(b) + " ");
} }
//[115, -34, -70, -67, 52, -65, -48, 64, 89, 60, 29, 61, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0] //[115, -34, -70, -67, 52, -65, -48, 64, 89, 60, 29, 61, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
this.globalX = ByteUtils.bytesToFloat(src, 0); this.globalX = ByteUtils.bytesToFloat(src, 0);

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.af;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/20 15:49
*/
public class PathStateSequence { public class PathStateSequence {
//src //src
private byte[] src; private byte[] src;
@ -14,6 +9,7 @@ public class PathStateSequence {
public Integer serialNumber; public Integer serialNumber;
//序列号 //序列号
public Integer pathId; public Integer pathId;
public PathStateSequence(byte[] src) { public PathStateSequence(byte[] src) {
this.src = src; this.src = src;
this.serialNumber = ByteUtils.bytesToInt(src, 0); this.serialNumber = ByteUtils.bytesToInt(src, 0);

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.af;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/20 15:49
*/
public class PointStateSequence { public class PointStateSequence {
//src //src
private byte[] src; private byte[] src;
@ -14,6 +9,7 @@ public class PointStateSequence {
public Integer serialNumber; public Integer serialNumber;
//序列号 //序列号
public Integer pointId; public Integer pointId;
public PointStateSequence(byte[] src) { public PointStateSequence(byte[] src) {
this.src = src; this.src = src;
this.serialNumber = ByteUtils.bytesToInt(src, 0); this.serialNumber = ByteUtils.bytesToInt(src, 0);

View File

@ -1,13 +1,9 @@
package org.opentcs.kc.udp.agv.param.function.af; package org.opentcs.kc.udp.agv.param.function.af;
import java.util.ArrayList;
import java.util.List; import java.util.List;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/20 14:55
*/
public class QueryRobotStatusRsp { public class QueryRobotStatusRsp {
private byte[] src; private byte[] src;
public byte abnormal_size; public byte abnormal_size;
@ -22,27 +18,46 @@ public class QueryRobotStatusRsp {
public List<ActionInfo> actionInfoList; public List<ActionInfo> actionInfoList;
public QueryRobotStatusRsp(byte[] src) { public QueryRobotStatusRsp(byte[] src) {
if (src == null) {
throw new RuntimeException("method_QueryRobotStatusRsp_params_src_is_NUll");
}
this.src = src; this.src = src;
this.abnormal_size = src[0]; this.abnormal_size = src[0];
this.action_size = src[1]; this.action_size = src[1];
this.remain = ByteUtils.copyBytes(src,2,2); //index 2 this.remain = ByteUtils.copyBytes(src, 2, 2); //index 2
this.locationStatusInfo = new LocationStatusInfo(ByteUtils.copyBytes(src,4,32)); this.locationStatusInfo = new LocationStatusInfo(ByteUtils.copyBytes(src, 4, 32));
this.runningStatusInfo = new RunningStatusInfo(ByteUtils.copyBytes(src,36,20)); this.runningStatusInfo = new RunningStatusInfo(ByteUtils.copyBytes(src, 36, 20));
Integer pointSize = ByteUtils.toInt(src[60]); Integer pointSize = ByteUtils.toInt(src[60]);
Integer edgeSize = ByteUtils.toInt(src[61]); Integer edgeSize = ByteUtils.toInt(src[61]);
Integer taskByteSize = 12+8*(pointSize+edgeSize); Integer taskByteSize = 12 + 8 * (pointSize + edgeSize);
this.taskStatusInfo = new TaskStatusInfo(ByteUtils.copyBytes(src,56,taskByteSize)); this.taskStatusInfo = new TaskStatusInfo(ByteUtils.copyBytes(src, 56, taskByteSize));
this.batteryStatusInfo = new BatteryStatusInfo(ByteUtils.copyBytes(src,56+taskByteSize,20)); this.batteryStatusInfo = new BatteryStatusInfo(ByteUtils.copyBytes(src, 56 + taskByteSize, 20));
if(this.abnormal_size>0){ if (this.abnormal_size > 0) {
for(int i=0;i<this.abnormal_size;i++){ if (this.abnormalEventStatusInfoList == null) {
this.abnormalEventStatusInfoList.add(new AbnormalEventStatusInfo(ByteUtils.copyBytes(src,56+taskByteSize+20+12*i,12))); this.abnormalEventStatusInfoList = new ArrayList<>();
}
for (int i = 0; i < this.abnormal_size; i++) {
this.abnormalEventStatusInfoList.add(
new AbnormalEventStatusInfo(
ByteUtils.copyBytes(src, 56 + taskByteSize + 20 + 12 * i, 12)
)
);
} }
} }
if(this.action_size>0){ if (this.action_size > 0) {
for(int i=0;i<this.action_size;i++){ if (this.actionInfoList == null) {
this.actionInfoList.add(new ActionInfo(ByteUtils.copyBytes(src,56+taskByteSize+20+12*this.abnormal_size+12*i,12))); this.actionInfoList = new ArrayList<>();
}
for (int i = 0; i < this.action_size; i++) {
this.actionInfoList.add(
new ActionInfo(
ByteUtils.copyBytes(
src, 56 + taskByteSize + 20 + 12 * this.abnormal_size + 12 * i, 12
)
)
);
} }
} }

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.af;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/20 15:22
*/
public class RunningStatusInfo { public class RunningStatusInfo {
private byte[] src; private byte[] src;
//4个字节, 轴x 速度,单位 m/s //4个字节, 轴x 速度,单位 m/s

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.af;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/20 15:22
*/
public class TaskStatusInfo { public class TaskStatusInfo {
private byte[] src; private byte[] src;
//订单 ID //订单 ID
@ -41,9 +36,10 @@ public class TaskStatusInfo {
if (edgeStatusNum > 0) { if (edgeStatusNum > 0) {
this.pathStatusInfo = new PathStateSequence[edgeStatusNum]; this.pathStatusInfo = new PathStateSequence[edgeStatusNum];
for (int i = 0; i < edgeStatusNum; i++) { for (int i = 0; i < edgeStatusNum; i++) {
this.pathStatusInfo[i] = new PathStateSequence(ByteUtils.copyBytes(src, 12 + pointStatusNum*8 + 8 * pointStatusNum + 8 * i, 8)); this.pathStatusInfo[i] = new PathStateSequence(
ByteUtils.copyBytes(src, 12 + pointStatusNum * 8 + 8 * pointStatusNum + 8 * i, 8)
);
} }
} }
} }
} }

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.b0;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/23 13:34
*/
public class QueryCargoStatusRsp { public class QueryCargoStatusRsp {
//src //src
private byte[] src; private byte[] src;

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.b1;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/23 14:30
*/
public class SubscribeInfo { public class SubscribeInfo {
//站控协议命令码2个字节 //站控协议命令码2个字节
public byte[] commandCode; public byte[] commandCode;
@ -20,7 +15,7 @@ public class SubscribeInfo {
public SubscribeInfo(byte[] commandCode, Short intervalTime, Integer durationTime) { public SubscribeInfo(byte[] commandCode, Short intervalTime, Integer durationTime) {
this.commandCode = commandCode; this.commandCode = commandCode;
this.intervalTime = ByteUtils.shortToBytes(intervalTime); this.intervalTime = ByteUtils.shortToBytes(intervalTime);
this.durationTime = ByteUtils.intToBytes(durationTime,4); this.durationTime = ByteUtils.intToBytes(durationTime, 4);
this.reserved = new byte[8]; this.reserved = new byte[8];
} }
} }

View File

@ -4,16 +4,12 @@ import java.security.SecureRandom;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/1/23 14:23
*/
public class SubscribeParam { public class SubscribeParam {
private byte[] src; private byte[] src;
private List<SubscribeInfo> subscribeInfoList; private List<SubscribeInfo> subscribeInfoList;
//uuid,64个字节 //uuid,64个字节
private byte[] uuid; private byte[] uuid;
public SubscribeParam(List<SubscribeInfo> subscribeInfoList) { public SubscribeParam(List<SubscribeInfo> subscribeInfoList) {
this.subscribeInfoList = subscribeInfoList; this.subscribeInfoList = subscribeInfoList;
this.uuid = generate64ByteUUID(); this.uuid = generate64ByteUUID();
@ -34,7 +30,7 @@ public class SubscribeParam {
return uuidBytes; return uuidBytes;
} }
public byte[] toBytes(){ public byte[] toBytes() {
src = new byte[192]; src = new byte[192];
List<Byte> bytes = new ArrayList<>(); List<Byte> bytes = new ArrayList<>();
for (SubscribeInfo subscribeInfo : subscribeInfoList) { for (SubscribeInfo subscribeInfo : subscribeInfoList) {
@ -60,7 +56,7 @@ public class SubscribeParam {
src[i] = bytes.get(i); src[i] = bytes.get(i);
} }
for (int i = 0; i < uuid.length; i++) { for (int i = 0; i < uuid.length; i++) {
src[i+128] = uuid[i]; src[i + 128] = uuid[i];
} }
return src; return src;

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.b1;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/5 14:21
*/
public class SubscribeRsp { public class SubscribeRsp {
//源数组 //源数组
private byte[] src; private byte[] src;
@ -16,17 +11,19 @@ public class SubscribeRsp {
private byte errCode; private byte errCode;
//reserved 3个字节 //reserved 3个字节
private byte[] reserved; private byte[] reserved;
public SubscribeRsp(byte[] src) { public SubscribeRsp(byte[] src) {
this.src = src; this.src = src;
this.uuid = ByteUtils.copyOfRange(src,0,64); this.uuid = ByteUtils.copyOfRange(src, 0, 64);
this.errCode = src[64]; this.errCode = src[64];
this.reserved = ByteUtils.copyOfRange(src,65,3); this.reserved = ByteUtils.copyOfRange(src, 65, 3);
} }
public boolean isOk(){ public boolean isOk() {
if(this.errCode==0){ if (this.errCode == 0) {
return true; return true;
}else { }
else {
return false; return false;
} }
} }

View File

@ -4,11 +4,6 @@ import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled; import io.netty.buffer.Unpooled;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/8 10:05
*/
public class Action { public class Action {
//动作类型,2个字节 //动作类型,2个字节
private byte[] actionType; private byte[] actionType;
@ -26,13 +21,15 @@ public class Action {
private byte[] paramContent; private byte[] paramContent;
public Action(byte actionType, byte actionParallel, Integer actionId, Integer paramSize, byte[] paramContent) { public Action(
this.actionType = new byte[]{actionType, (byte)0x00}; byte actionType, byte actionParallel, Integer actionId, Integer paramSize, byte[] paramContent
) {
this.actionType = new byte[]{actionType, (byte) 0x00};
this.actionParallel = actionParallel; this.actionParallel = actionParallel;
this.actionRetain = (byte)0x00; this.actionRetain = (byte) 0x00;
this.actionId = ByteUtils.intToBytes(actionId,1); this.actionId = ByteUtils.intToBytes(actionId, 1);
this.paramSize = ByteUtils.usintTo1Byte(paramSize); this.paramSize = ByteUtils.usintTo1Byte(paramSize);
this.paramRetain2 = new byte[]{(byte)0x00, (byte)0x00,(byte)0x00}; this.paramRetain2 = new byte[]{(byte) 0x00, (byte) 0x00, (byte) 0x00};
this.paramContent = paramContent; this.paramContent = paramContent;
} }

View File

@ -4,11 +4,6 @@ import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled; import io.netty.buffer.Unpooled;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/10 8:56
*/
public class ActionSet { public class ActionSet {
public static void main(String[] args) { public static void main(String[] args) {
byte[] stop = stop0x01(1, (byte) 0x01); byte[] stop = stop0x01(1, (byte) 0x01);
@ -29,25 +24,25 @@ public class ActionSet {
public static Integer trayElevation0x16_paramsize = 4; public static Integer trayElevation0x16_paramsize = 4;
public static byte[] stop0x01(Integer orderId,byte isStopImmediately){ public static byte[] stop0x01(Integer orderId, byte isStopImmediately) {
ByteBuf byteBuf = Unpooled.buffer(8); ByteBuf byteBuf = Unpooled.buffer(8);
byteBuf.writeBytes(ByteUtils.intToBytes(orderId,1)); byteBuf.writeBytes(ByteUtils.intToBytes(orderId, 1));
byteBuf.writeByte(isStopImmediately); byteBuf.writeByte(isStopImmediately);
return byteBuf.array(); return byteBuf.array();
} }
public static byte[] recover0x02(Integer orderId,Integer taskKey){ public static byte[] recover0x02(Integer orderId, Integer taskKey) {
ByteBuf byteBuf = Unpooled.buffer(8); ByteBuf byteBuf = Unpooled.buffer(8);
byteBuf.writeBytes(ByteUtils.intToBytes(orderId,1)); byteBuf.writeBytes(ByteUtils.intToBytes(orderId, 1));
byteBuf.writeBytes(ByteUtils.intToBytes(taskKey,1)); byteBuf.writeBytes(ByteUtils.intToBytes(taskKey, 1));
return byteBuf.array(); return byteBuf.array();
} }
//取消任务 //取消任务
public static byte[] cancelTask0x03(Integer orderId,byte isStopImmediately){ public static byte[] cancelTask0x03(Integer orderId, byte isStopImmediately) {
ByteBuf byteBuf = Unpooled.buffer(8); ByteBuf byteBuf = Unpooled.buffer(8);
byteBuf.writeBytes(ByteUtils.intToBytes(orderId,1)); byteBuf.writeBytes(ByteUtils.intToBytes(orderId, 1));
byteBuf.writeByte(isStopImmediately); byteBuf.writeByte(isStopImmediately);
return byteBuf.array(); return byteBuf.array();
} }
@ -56,9 +51,11 @@ public class ActionSet {
* desc:叉齿升降 * desc:叉齿升降
* 支持的命令0xAE 0xB2 * 支持的命令0xAE 0xB2
* *
* */ */
public static byte[] forkliftElevation0x12(float lifitingHeight,byte hightMean,byte moveType, byte taskOperationType){ public static byte[] forkliftElevation0x12(
float lifitingHeight, byte hightMean, byte moveType, byte taskOperationType
) {
ByteBuf byteBuf = Unpooled.buffer(8); ByteBuf byteBuf = Unpooled.buffer(8);
byteBuf.writeBytes(ByteUtils.floatToBytes(lifitingHeight)); byteBuf.writeBytes(ByteUtils.floatToBytes(lifitingHeight));
byteBuf.writeByte(hightMean); byteBuf.writeByte(hightMean);
@ -68,7 +65,7 @@ public class ActionSet {
} }
//托盘升降 //托盘升降
public static byte[] trayElevation0x16(byte palletMovementMode){ public static byte[] trayElevation0x16(byte palletMovementMode) {
ByteBuf byteBuf = Unpooled.buffer(4); ByteBuf byteBuf = Unpooled.buffer(4);
byteBuf.writeByte(palletMovementMode); byteBuf.writeByte(palletMovementMode);
return byteBuf.array(); return byteBuf.array();

View File

@ -4,11 +4,6 @@ import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled; import io.netty.buffer.Unpooled;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: "0xAE 导航参数"
* @Author: caixiang
* @DATE: 2024/12/30 15:44
*/
public class NavigationParam { public class NavigationParam {
//订单 ID,4个字节 ,Integer,从1开始依次累加 //订单 ID,4个字节 ,Integer,从1开始依次累加
private byte[] orderId; private byte[] orderId;
@ -25,12 +20,14 @@ public class NavigationParam {
//任务中路径信息结构体长度 pathNum //任务中路径信息结构体长度 pathNum
private Path[] paths; private Path[] paths;
public NavigationParam(Integer orderId, Integer taskKey, byte pointNum, byte pathNum, Point[] points, Path[] paths) { public NavigationParam(
this.orderId = ByteUtils.intToBytes(orderId,1); Integer orderId, Integer taskKey, byte pointNum, byte pathNum, Point[] points, Path[] paths
this.taskKey = ByteUtils.intToBytes(taskKey,1); ) {
this.orderId = ByteUtils.intToBytes(orderId, 1);
this.taskKey = ByteUtils.intToBytes(taskKey, 1);
this.pointNum = pointNum; this.pointNum = pointNum;
this.pathNum = pathNum; this.pathNum = pathNum;
this.retain = new byte[]{(byte)0x00, (byte)0x00}; this.retain = new byte[]{(byte) 0x00, (byte) 0x00};
this.points = points; this.points = points;
this.paths = paths; this.paths = paths;
} }

View File

@ -4,11 +4,6 @@ import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled; import io.netty.buffer.Unpooled;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/8 9:39
*/
public class Path { public class Path {
//序列号,4个字节,Integer,用于定位段在整个任务中的位置 1 开始奇数递增 135......point 2 4 6这样path 和point就可以结合起来了以区分同一个段 ID 在任务中是否多次出现 //序列号,4个字节,Integer,用于定位段在整个任务中的位置 1 开始奇数递增 135......point 2 4 6这样path 和point就可以结合起来了以区分同一个段 ID 在任务中是否多次出现
private byte[] serialNum; private byte[] serialNum;
@ -36,17 +31,20 @@ public class Path {
private Action[] actions; private Action[] actions;
public Path(Integer serialNum, Integer pathId, float angle, byte isSpecifyAngle, byte drivePose, byte pathActionSize, float maxSpeed, float maxAngularSpeed, Action[] actions) { public Path(
this.serialNum = ByteUtils.intToBytes(serialNum,1); Integer serialNum, Integer pathId, float angle, byte isSpecifyAngle, byte drivePose,
this.pathId = ByteUtils.intToBytes(pathId,1); byte pathActionSize, float maxSpeed, float maxAngularSpeed, Action[] actions
) {
this.serialNum = ByteUtils.intToBytes(serialNum, 1);
this.pathId = ByteUtils.intToBytes(pathId, 1);
this.angle = ByteUtils.floatToBytes(angle); this.angle = ByteUtils.floatToBytes(angle);
this.isSpecifyAngle = isSpecifyAngle; this.isSpecifyAngle = isSpecifyAngle;
this.drivePose = drivePose; this.drivePose = drivePose;
this.pathActionSize = pathActionSize; this.pathActionSize = pathActionSize;
this.pathRetain = (byte)0x00; this.pathRetain = (byte) 0x00;
this.maxSpeed = ByteUtils.floatToBytes(maxSpeed); this.maxSpeed = ByteUtils.floatToBytes(maxSpeed);
this.maxAngularSpeed = ByteUtils.floatToBytes(maxAngularSpeed); this.maxAngularSpeed = ByteUtils.floatToBytes(maxAngularSpeed);
this.pathRetain2 = new byte[]{(byte)0x00, (byte)0x00,(byte)0x00,(byte)0x00}; this.pathRetain2 = new byte[]{(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00};
this.actions = actions; this.actions = actions;
} }
@ -62,9 +60,11 @@ public class Path {
byteBuf.writeBytes(maxSpeed); byteBuf.writeBytes(maxSpeed);
byteBuf.writeBytes(maxAngularSpeed); byteBuf.writeBytes(maxAngularSpeed);
byteBuf.writeBytes(pathRetain2); byteBuf.writeBytes(pathRetain2);
if (actions != null) {
for (Action action : actions) { for (Action action : actions) {
byteBuf.writeBytes(action.toBytes()); byteBuf.writeBytes(action.toBytes());
} }
}
int i = byteBuf.writerIndex(); int i = byteBuf.writerIndex();
byte[] bytes = new byte[i]; byte[] bytes = new byte[i];
byteBuf.readBytes(bytes); byteBuf.readBytes(bytes);

View File

@ -4,11 +4,6 @@ import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled; import io.netty.buffer.Unpooled;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/8 9:39
*/
public class Point { public class Point {
//序列号,4个字节,Integer,用于定位点在整个任务中的位置 0 开始偶数递增 0246......以区分同一个点 ID 在任务中是否多次出现 //序列号,4个字节,Integer,用于定位点在整个任务中的位置 0 开始偶数递增 0246......以区分同一个点 ID 在任务中是否多次出现
private byte[] serialNum; private byte[] serialNum;
@ -25,13 +20,17 @@ public class Point {
//任务中点上动作结构体,这里数组的长度是 pointActionSize 长度 //任务中点上动作结构体,这里数组的长度是 pointActionSize 长度
private Action[] actions; private Action[] actions;
public Point(Integer serialNum, Integer pointId, float angle, byte isSpecifyAngle, byte pointActionSize, Action[] actions) { public Point(
this.serialNum = ByteUtils.intToBytes(serialNum,1); Integer serialNum, Integer pointId, float angle, byte isSpecifyAngle, byte pointActionSize,
this.pointId = ByteUtils.intToBytes(pointId,1); Action[] actions
) {
this.serialNum = ByteUtils.intToBytes(serialNum, 1);
this.pointId = ByteUtils.intToBytes(pointId, 1);
this.angle = ByteUtils.floatToBytes(angle); this.angle = ByteUtils.floatToBytes(angle);
this.isSpecifyAngle = isSpecifyAngle; this.isSpecifyAngle = isSpecifyAngle;
this.pointActionSize = pointActionSize; this.pointActionSize = pointActionSize;
this.pointRetain = new byte[]{(byte)0x00, (byte)0x00,(byte)0x00,(byte)0x00,(byte)0x00,(byte)0x00}; this.pointRetain = new byte[]{(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
(byte) 0x00};
this.actions = actions; this.actions = actions;
} }
@ -43,9 +42,11 @@ public class Point {
byteBuf.writeByte(isSpecifyAngle); byteBuf.writeByte(isSpecifyAngle);
byteBuf.writeByte(pointActionSize); byteBuf.writeByte(pointActionSize);
byteBuf.writeBytes(pointRetain); byteBuf.writeBytes(pointRetain);
if (actions != null) {
for (Action action : actions) { for (Action action : actions) {
byteBuf.writeBytes(action.toBytes()); byteBuf.writeBytes(action.toBytes());
} }
}
int i = byteBuf.writerIndex(); int i = byteBuf.writerIndex();
byte[] bytes = new byte[i]; byte[] bytes = new byte[i];
byteBuf.readBytes(bytes); byteBuf.readBytes(bytes);

View File

@ -4,22 +4,18 @@ import java.util.ArrayList;
import java.util.List; import java.util.List;
import org.opentcs.kc.common.byteutils.ByteUtil; import org.opentcs.kc.common.byteutils.ByteUtil;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2024/12/30 15:44
*/
public class ReadParam { public class ReadParam {
private byte valueNum; private byte valueNum;
private byte[] retain; private byte[] retain;
private byte[] valueId; private byte[] valueId;
private byte[] readStrValues; private byte[] readStrValues;
public ReadParam(byte[] valueId, List<ReadStrValue> rvalues) { public ReadParam(byte[] valueId, List<ReadStrValue> rvalues) {
// 这里的valueId 也是 header里面的 transationId // 这里的valueId 也是 header里面的 transationId
valueNum = ByteUtil.intTo1Byte(rvalues.size()); valueNum = ByteUtil.intTo1Byte(rvalues.size());
retain = new byte[]{0x00,0x00,0x00}; retain = new byte[]{0x00, 0x00, 0x00};
this.valueId = new byte[]{valueId[0],valueId[1],0x00,0x00}; this.valueId = new byte[]{valueId[0], valueId[1], 0x00, 0x00};
List<Byte> bytes = new ArrayList<>(); List<Byte> bytes = new ArrayList<>();
for (ReadStrValue b : rvalues) { for (ReadStrValue b : rvalues) {
bytes.addAll(b.toBytes()); bytes.addAll(b.toBytes());

View File

@ -2,11 +2,6 @@ package org.opentcs.kc.udp.agv.param.function.read;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/5 14:55
*/
public class ReadRsp { public class ReadRsp {
//源数组 //源数组
private byte[] src; private byte[] src;
@ -22,15 +17,16 @@ public class ReadRsp {
public ReadRsp(byte[] src) { public ReadRsp(byte[] src) {
this.src = src; this.src = src;
this.valueId = ByteUtils.copyBytes(src, 0, 4); this.valueId = ByteUtils.copyBytes(src, 0, 4);
this.valueByteLength = ByteUtils.bytesToShort(ByteUtils.copyBytes(src, 4, 2),1); this.valueByteLength = ByteUtils.bytesToShort(ByteUtils.copyBytes(src, 4, 2), 1);
this.reserved = ByteUtils.copyBytes(src, 6, 2); this.reserved = ByteUtils.copyBytes(src, 6, 2);
this.dataValue = ByteUtils.copyBytes(src, 8, valueByteLength-8); this.dataValue = ByteUtils.copyBytes(src, 8, valueByteLength - 8);
} }
public boolean isOk(){ public boolean isOk() {
if(valueByteLength<=0){ if (valueByteLength <= 0) {
return false; return false;
}else { }
else {
return true; return true;
} }
} }

View File

@ -4,17 +4,13 @@ import java.util.ArrayList;
import java.util.List; import java.util.List;
import org.opentcs.kc.common.byteutils.ByteUtil; import org.opentcs.kc.common.byteutils.ByteUtil;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2024/12/30 15:46
*/
public class ReadStrValue { public class ReadStrValue {
//变量名16个字节( 以这个变量名来定位变量的) //变量名16个字节( 以这个变量名来定位变量的)
private byte[] varName; private byte[] varName;
//成员变量数量4个字节 //成员变量数量4个字节
private byte[] memberVarNum; private byte[] memberVarNum;
private byte[] memberList; private byte[] memberList;
public ReadStrValue(String varName, Integer memberVarNum, List<ReadValueMember> memberList) { public ReadStrValue(String varName, Integer memberVarNum, List<ReadValueMember> memberList) {
this.varName = ByteUtil.stringTo16Byte(varName); this.varName = ByteUtil.stringTo16Byte(varName);
this.memberVarNum = ByteUtil.intToBytes(memberVarNum); this.memberVarNum = ByteUtil.intToBytes(memberVarNum);
@ -27,6 +23,7 @@ public class ReadStrValue {
this.memberList[i] = bytes.get(i); this.memberList[i] = bytes.get(i);
} }
} }
public ReadStrValue(String varName) { public ReadStrValue(String varName) {
this.varName = ByteUtil.stringTo16Byte(varName); this.varName = ByteUtil.stringTo16Byte(varName);
this.memberVarNum = ByteUtil.intToBytes(0); this.memberVarNum = ByteUtil.intToBytes(0);

View File

@ -4,11 +4,6 @@ import java.util.ArrayList;
import java.util.List; import java.util.List;
import org.opentcs.kc.common.byteutils.ByteUtil; import org.opentcs.kc.common.byteutils.ByteUtil;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2024/12/30 15:49
*/
public class ReadValueMember { public class ReadValueMember {
//偏移值就是以这个变量的起始点为基准偏移几个字节来读取数组里的后面几个变量基本上用于数组变量单体变量用不着 2个字节 //偏移值就是以这个变量的起始点为基准偏移几个字节来读取数组里的后面几个变量基本上用于数组变量单体变量用不着 2个字节
private byte[] offsetValue; private byte[] offsetValue;

View File

@ -5,11 +5,6 @@ import io.netty.buffer.Unpooled;
import java.util.List; import java.util.List;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/5 16:41
*/
public class WriteParam { public class WriteParam {
//变量数量 1个字节 ,最大只支持 15 个变量 //变量数量 1个字节 ,最大只支持 15 个变量
private byte valueNum; private byte valueNum;
@ -17,9 +12,10 @@ public class WriteParam {
private byte[] retain; private byte[] retain;
//这里的length 就是 valueNum //这里的length 就是 valueNum
private List<WriteStrValue> values; private List<WriteStrValue> values;
public WriteParam(Integer valueNum, List<WriteStrValue> values) { public WriteParam(Integer valueNum, List<WriteStrValue> values) {
this.valueNum = ByteUtils.usintTo1Byte(valueNum); this.valueNum = ByteUtils.usintTo1Byte(valueNum);
this.retain = new byte[]{(byte)0x00,(byte)0x00,(byte)0x00}; this.retain = new byte[]{(byte) 0x00, (byte) 0x00, (byte) 0x00};
this.values = values; this.values = values;
} }

View File

@ -5,11 +5,6 @@ import io.netty.buffer.Unpooled;
import java.util.List; import java.util.List;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/5 16:43
*/
public class WriteStrValue { public class WriteStrValue {
//变量名 16个字节 //变量名 16个字节
private String valueName; private String valueName;
@ -26,7 +21,7 @@ public class WriteStrValue {
public byte[] toBytes() { public byte[] toBytes() {
ByteBuf byteBuf = Unpooled.buffer(); ByteBuf byteBuf = Unpooled.buffer();
byteBuf.writeBytes(ByteUtils.stringToBytes(valueName, 16)); byteBuf.writeBytes(ByteUtils.stringToBytes(valueName, 16));
byteBuf.writeBytes(ByteUtils.intToBytes(valueNum,1)); byteBuf.writeBytes(ByteUtils.intToBytes(valueNum, 1));
for (WriteValueMember member : members) { for (WriteValueMember member : members) {
byteBuf.writeBytes(member.toBytes()); byteBuf.writeBytes(member.toBytes());
} }

View File

@ -4,11 +4,6 @@ import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled; import io.netty.buffer.Unpooled;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/5 16:44
*/
public class WriteValueMember { public class WriteValueMember {
//变量成员偏移 2个字节 //变量成员偏移 2个字节
private byte[] valueOffset; private byte[] valueOffset;
@ -16,6 +11,7 @@ public class WriteValueMember {
private byte[] valueLength; private byte[] valueLength;
//变量成员值 4个字节 //变量成员值 4个字节
private byte[] newValue; private byte[] newValue;
public WriteValueMember(Short valueOffset, Short valueLength, byte[] newValue) { public WriteValueMember(Short valueOffset, Short valueLength, byte[] newValue) {
this.valueOffset = ByteUtils.shortToBytes(valueOffset); this.valueOffset = ByteUtils.shortToBytes(valueOffset);
this.valueLength = ByteUtils.shortToBytes(valueLength); this.valueLength = ByteUtils.shortToBytes(valueLength);

View File

@ -4,11 +4,6 @@ import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled; import io.netty.buffer.Unpooled;
import org.opentcs.kc.common.byteutils.ByteUtil; import org.opentcs.kc.common.byteutils.ByteUtil;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/7 9:21
*/
public class RobotSetPosition { public class RobotSetPosition {
//机器人 x 坐标 ,8 个字节 //机器人 x 坐标 ,8 个字节
private byte[] robotX; private byte[] robotX;
@ -22,6 +17,7 @@ public class RobotSetPosition {
this.robotY = ByteUtil.doubleToBytes(robotY); this.robotY = ByteUtil.doubleToBytes(robotY);
this.robotAngle = ByteUtil.doubleToBytes(robotAngle); this.robotAngle = ByteUtil.doubleToBytes(robotAngle);
} }
public byte[] toBytes() { public byte[] toBytes() {
ByteBuf byteBuf = Unpooled.buffer(24); ByteBuf byteBuf = Unpooled.buffer(24);
byteBuf.writeBytes(robotX); byteBuf.writeBytes(robotX);

View File

@ -3,11 +3,6 @@ package org.opentcs.kc.udp.agv.param.function.x17;
import java.util.Arrays; import java.util.Arrays;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2025/2/7 9:43
*/
public class QueryRobotRunStatusRsp { public class QueryRobotRunStatusRsp {
//本体温度 ,8个字节 ,double //本体温度 ,8个字节 ,double
private double temp; private double temp;

View File

@ -4,11 +4,6 @@ package org.opentcs.kc.udp.agv.param.rsp;
import org.opentcs.kc.common.byteutils.ByteUtils; import org.opentcs.kc.common.byteutils.ByteUtils;
import org.opentcs.kc.udp.agv.param.AgvEventHeader; import org.opentcs.kc.udp.agv.param.AgvEventHeader;
/**
* @Desc: "通用的数据传输底层类"
* @Author: caixiang
* @DATE: 2022/10/18 16:22
*/
public class RcvEventPackage { public class RcvEventPackage {
private boolean isOk; private boolean isOk;
@ -20,37 +15,46 @@ public class RcvEventPackage {
private String getContent(byte i) { private String getContent(byte i) {
if(i==0x00){ if (i == 0x00) {
return "成功执行"; return "成功执行";
}else if(i==0x01){ }
else if (i == 0x01) {
return "执行失败,原因未知"; return "执行失败,原因未知";
}else if(i==0x02){ }
else if (i == 0x02) {
return "服务码错误"; return "服务码错误";
}else if(i==0x03){ }
else if (i == 0x03) {
return "命令码错误"; return "命令码错误";
}else if(i==0x04){ }
else if (i == 0x04) {
return "报文头部错误"; return "报文头部错误";
}else if(i==0x80){ }
else if (i == 0x80) {
return "无法执行命令,因为当前车辆导航状态与命令冲突"; return "无法执行命令,因为当前车辆导航状态与命令冲突";
}else if(i==0xFF){ }
else if (i == 0xFF) {
return "协议授权码错误"; return "协议授权码错误";
}else { }
else {
return "未知错误"; return "未知错误";
} }
} }
public RcvEventPackage(byte executionCode, byte[] value) { public RcvEventPackage(byte executionCode, byte[] value) {
if(executionCode == 0x00){ if (executionCode == 0x00) {
this.isOk = true; this.isOk = true;
this.header = new AgvEventHeader(ByteUtils.copyBytes(value,0,28)); this.header = new AgvEventHeader(ByteUtils.copyBytes(value, 0, 28));
this.dataBytes = ByteUtils.copyBytes(value,28,value.length-28); this.dataBytes = ByteUtils.copyBytes(value, 28, value.length - 28);
}else { }
else {
this.isOk = false; this.isOk = false;
} }
this.value = value; this.value = value;
this.content = getContent(executionCode); this.content = getContent(executionCode);
} }
public RcvEventPackage() { public RcvEventPackage() {
} }

View File

@ -9,6 +9,7 @@ import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.DatagramPacket; import io.netty.channel.socket.DatagramPacket;
import io.netty.channel.socket.nio.NioDatagramChannel; import io.netty.channel.socket.nio.NioDatagramChannel;
import java.net.InetSocketAddress; import java.net.InetSocketAddress;
import java.util.HashMap;
import java.util.Set; import java.util.Set;
import org.opentcs.access.KernelServicePortal; import org.opentcs.access.KernelServicePortal;
import org.opentcs.access.rmi.KernelServicePortalBuilder; import org.opentcs.access.rmi.KernelServicePortalBuilder;
@ -27,21 +28,17 @@ import org.opentcs.kc.udp.agv.param.rsp.RcvEventPackage;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
/**
* @Desc: ""
* @Author: caixiang
* @DATE: 2022/1/15 13:01
*/
public enum UDPClient { public enum UDPClient {
//如果要配置多个链接 local1 local2 .... 这样排下去好了 //如果要配置多个链接 local1 local2 .... 这样排下去好了
localAGV("1","192.168.0.211",17804,55678), localAGV("50", "192.168.124.124", 17804, 55678),
//local("127.0.0.1",502,true),
; ;
// 服务端地址+端口 // 服务端用户名+密码+地址+端口
private String SERVICE_HOST = "192.168.0.123"; private String SERVICE_USER = GuestUserCredentials.USER;
private Integer SERVICE_PORT = 1099; private String SERVICE_PWD = GuestUserCredentials.PASSWORD;
private String SERVICE_HOST = GuestUserCredentials.IP;
private Integer SERVICE_PORT = GuestUserCredentials.PORT;
private String name; private String name;
private String host; private String host;
@ -55,9 +52,8 @@ public enum UDPClient {
private Channel conn; private Channel conn;
private boolean isOnline; private boolean isOnline;
private static final Logger logger = LoggerFactory.getLogger("AGVLOGGER"); // private static final Logger logger = LoggerFactory.getLogger("AGVLOGGER");
private static final Logger logger = LoggerFactory.getLogger(UDPClient.class);
// ==== 连接节点配置信息 ===== 开始 // ==== 连接节点配置信息 ===== 开始
@ -77,29 +73,33 @@ public enum UDPClient {
} }
public String getHost() {
public String getHost(){
return this.host; return this.host;
} }
public String getName(){
public String getName() {
return this.name; return this.name;
} }
public void setIsOnline(boolean isOnline){
public void setIsOnline(boolean isOnline) {
this.isOnline = isOnline; this.isOnline = isOnline;
} }
/** /**
* desc : 判断此链接 健康状况 * desc : 判断此链接 健康状况
* return * return
* true : 此tcp连接 正常 * true : 此tcp连接 正常
* false : 此tcp连接 异常 * false : 此tcp连接 异常
* */ */
public boolean isOnline(){ public boolean isOnline() {
return isOnline; return isOnline;
} }
public void close(){
public void close() {
//手动关闭连接会出发InActivite 事件 //手动关闭连接会出发InActivite 事件
group.shutdownGracefully(); group.shutdownGracefully();
} }
public static ByteBuf byteArrayToByteBuf(byte[] byteArray) { public static ByteBuf byteArrayToByteBuf(byte[] byteArray) {
// 使用Unpooled类创建ByteBuf // 使用Unpooled类创建ByteBuf
ByteBuf byteBuf = Unpooled.buffer(byteArray.length); ByteBuf byteBuf = Unpooled.buffer(byteArray.length);
@ -107,130 +107,143 @@ public enum UDPClient {
byteBuf.writeBytes(byteArray); byteBuf.writeBytes(byteArray);
return byteBuf; return byteBuf;
} }
/** /**
* --线程安全的-- * --线程安全的--
*
* *
* 如果返回null就代表出现了异常并且尝试了 retryMax 次数并且尝试重置连接 * 如果返回null就代表出现了异常并且尝试了 retryMax 次数并且尝试重置连接
* */ */
public RcvEventPackage send(AgvEvent event) { public RcvEventPackage send(AgvEvent event) {
try { try {
//sendread 报文 //sendread 报文
//应该是ReadRequestFrame 继承 Packet 然后直接 Tio.bSend(this.conn, ReadRequestFrame) //应该是ReadRequestFrame 继承 Packet 然后直接 Tio.bSend(this.conn, ReadRequestFrame)
Package mbPackage = event.toBytes(); Package mbPackage = event.toBytes();
AsyncFuture<Package> add = SendedList.add(mbPackage.getTransationId(),null); AsyncFuture<Package> add = SendedList.add(mbPackage.getTransationId(), null);
this.conn.writeAndFlush( this.conn.writeAndFlush(
new DatagramPacket( new DatagramPacket(
byteArrayToByteBuf(mbPackage.getBody()), byteArrayToByteBuf(mbPackage.getBody()),
new InetSocketAddress(this.host,this.port))) new InetSocketAddress(this.host, this.port)
)
)
.sync(); .sync();
Package aPackage = add.get(5000L, mbPackage.getTransationId()); Package aPackage = add.get(5000L, mbPackage.getTransationId());
byte[] body = aPackage.getBody(); byte[] body = aPackage.getBody();
String errMsg = " [ AGVclient - send success ] [ "+name+" host: "+ this.host +" ]"+event.toString(); String errMsg = " [ AGVclient - send success ] [ " + name + " host: " + this.host
+ " ]" + event.toString();
logger.info(errMsg); logger.info(errMsg);
//注意这里的body 是整个 response结构包括 : 授权码 + header + body //注意这里的body 是整个 response结构包括 : 授权码 + header + body
return new RcvEventPackage(body[22],body); return new RcvEventPackage(body[22], body);
}catch (Throwable e) { }
catch (Throwable e) {
//e.printStackTrace(); //e.printStackTrace();
String errMsg = " [ AGVclient - Read ] [ "+name+" host: "+ this.host +" ] ( occur err ) errTime: "+" ; send errMsg : "+e.getMessage()+" ; event :"+event.toString(); String errMsg = " [ AGVclient - Read ] [ " + name + " host: " + this.host
+ " ] ( occur err ) errTime: " + " ; send errMsg : " + e.getMessage()
+ " ; event :" + event.toString();
logger.info(errMsg); logger.info(errMsg);
throw new RuntimeException(errMsg); throw new RuntimeException(errMsg);
} }
} }
public void subscribe0xB0(RcvEventPackage rcv){ public void subscribeKC(RcvEventPackage rcv, byte[] body) {
if (rcv.isOk()) {
KernelServicePortal servicePortal = new KernelServicePortalBuilder(
SERVICE_USER, SERVICE_PWD
).build();
servicePortal.login(SERVICE_HOST, SERVICE_PORT);
VehicleService vehicleService = servicePortal.getVehicleService();
Vehicle vehicle = vehicleService.fetchObject(Vehicle.class, name);
vehicleService.sendCommAdapterMessage(vehicle.getReference(), body);
servicePortal.logout();
}
else {
System.out.println();
System.out.println("subscribe received transationId : " + "isok:" + rcv.isOk());
}
}
public void subscribe0xB0(RcvEventPackage rcv, byte[] body) {
if (rcv.isOk()) { if (rcv.isOk()) {
this.achieveSub0xB0(body);
if(name.equals("1")){
this.achieveSub0xB0(name, rcv);
}else if(name.equals("2")){
this.achieveSub0xB0(name, rcv);
} }
else {
} else {
System.out.println(); System.out.println();
System.out.println("subscribe0xB0 received transationId : "+ "isok:"+rcv.isOk()); System.out.println("subscribe0xB0 received transationId : " + "isok:" + rcv.isOk());
} }
} }
public void subscribe0xAF(RcvEventPackage rcv){
public void subscribe0xAF(RcvEventPackage rcv, byte[] body) {
if (rcv.isOk()) { if (rcv.isOk()) {
this.achieveSub0xAF(body);
if (name.equals("1")) {
this.achieveSub0xAF(name, rcv);
} else if (name.equals("2")) {
this.achieveSub0xAF(name, rcv);
} }
else {
} else {
System.out.println(); System.out.println();
System.out.println("subscribe0xAF received transationId : "+ "isok:"+rcv.isOk()); System.out.println("subscribe0xAF received transationId : " + "isok:" + rcv.isOk());
} }
} }
/** /**
* 0xAF上报信息传入通讯适配器 * 0xAF上报信息传入通讯适配器
* @param vehicleName 车辆名称 *
* @param rcv 响应数据 * @param body 响应数据
*/ */
private void achieveSub0xAF(String vehicleName ,RcvEventPackage rcv){ private void achieveSub0xAF(byte[] body) {
// QueryRobotStatusRsp queryRobotStatusRsp = new QueryRobotStatusRsp(rcv.getDataBytes());
QueryRobotStatusRsp queryRobotStatusRsp = new QueryRobotStatusRsp(rcv.getDataBytes()); KernelServicePortal servicePortal = new KernelServicePortalBuilder(
SERVICE_USER, SERVICE_PWD
System.out.println(); ).build();
System.out.println("received subscribe 0xAF List : "+ "isok:"+rcv.isOk());
for (byte b:rcv.getValue()){
System.out.print(byteToHex(b)+" ");
}
KernelServicePortal servicePortal = new KernelServicePortalBuilder(GuestUserCredentials.USER, GuestUserCredentials.PASSWORD).build();
servicePortal.login(SERVICE_HOST, SERVICE_PORT); servicePortal.login(SERVICE_HOST, SERVICE_PORT);
VehicleService vehicleService = servicePortal.getVehicleService(); VehicleService vehicleService = servicePortal.getVehicleService();
Vehicle vehicle = vehicleService.fetchObject(Vehicle.class, vehicleName); Vehicle vehicle = vehicleService.fetchObject(Vehicle.class, name);
//将AGV控制器上报信息传入通讯适配器 vehicleService.sendCommAdapterMessage(vehicle.getReference(), body);
vehicleService.sendCommAdapterMessage(vehicle.getReference(), queryRobotStatusRsp);
servicePortal.logout();
} }
/** /**
* 0xB0上报信息传入通讯适配器 * 0xB0上报信息传入通讯适配器
* @param vehicleName 车辆名称 *
* @param rcv 响应数据 * @param body 响应数据
*/ */
private void achieveSub0xB0(String vehicleName ,RcvEventPackage rcv){ private void achieveSub0xB0(byte[] body) {
QueryCargoStatusRsp queryCargoStatusRsp = new QueryCargoStatusRsp(rcv.getDataBytes()); // QueryCargoStatusRsp queryCargoStatusRsp = new QueryCargoStatusRsp(rcv.getDataBytes());
System.out.println(); // System.out.println();
System.out.println("received subscribe 0xB0 List : "+ "isok:"+rcv.isOk()); // System.out.println("received subscribe 0xB0 List : " + "isok:" + rcv.isOk());
for (byte b:rcv.getValue()){ // for (byte b : rcv.getValue()) {
System.out.print(byteToHex(b)+" "); // System.out.print(byteToHex(b) + " ");
} // }
KernelServicePortal servicePortal = new KernelServicePortalBuilder(GuestUserCredentials.USER, GuestUserCredentials.PASSWORD).build(); KernelServicePortal servicePortal = new KernelServicePortalBuilder(
SERVICE_USER, SERVICE_PWD
).build();
servicePortal.login(SERVICE_HOST, SERVICE_PORT); servicePortal.login(SERVICE_HOST, SERVICE_PORT);
VehicleService vehicleService = servicePortal.getVehicleService(); VehicleService vehicleService = servicePortal.getVehicleService();
Vehicle vehicle = vehicleService.fetchObject(Vehicle.class, vehicleName); Vehicle vehicle = vehicleService.fetchObject(Vehicle.class, name);
if (vehicle == null) {
throw new RuntimeException("vehicle name:" + name + ",does not exist");
}
//将AGV控制器上报信息传入通讯适配器 //将AGV控制器上报信息传入通讯适配器
vehicleService.sendCommAdapterMessage(vehicle.getReference(), queryCargoStatusRsp); vehicleService.sendCommAdapterMessage(vehicle.getReference(), body);
servicePortal.logout(); servicePortal.logout();
} }
public static void printInfo(AgvEvent agvEvent){ public static void printInfo(AgvEvent agvEvent) {
System.out.println("sended transationId : "+agvEvent.getTransationIdString()); System.out.println("sended transationId : " + agvEvent.getTransationIdString());
for (byte b:agvEvent.toBytes().getBody()){ for (byte b : agvEvent.toBytes().getBody()) {
System.out.print(byteToHex(b)+" "); System.out.print(byteToHex(b) + " ");
} }
} }
public static String byteToHex(byte b) { public static String byteToHex(byte b) {
// 将byte转换为无符号整数 // 将byte转换为无符号整数
int unsignedByte = b & 0xFF; int unsignedByte = b & 0xFF;
@ -242,19 +255,21 @@ public enum UDPClient {
} }
return hexString; return hexString;
} }
/** /**
* decs: 在项目启动的时候初始化 后面就不会初始化了 * decs: 在项目启动的时候初始化 后面就不会初始化了
* */ */
private void initClient() { private void initClient() {
//NioEventLoopGroup Bootstrap 这些资源其实是不用 release的因为全局共用一份的你释放了 下次还得再new //NioEventLoopGroup Bootstrap 这些资源其实是不用 release的因为全局共用一份的你释放了 下次还得再new
NioEventLoopGroup group = new NioEventLoopGroup(); NioEventLoopGroup group = new NioEventLoopGroup();
this.bootstrap = new Bootstrap(); this.bootstrap = new Bootstrap();
this.bootstrap.group(group) this.bootstrap.group(group)
/*由于我们用的是UDP协议所以要用NioDatagramChannel来创建*/ /* 由于我们用的是UDP协议所以要用NioDatagramChannel来创建 */
.channel(NioDatagramChannel.class) .channel(NioDatagramChannel.class)
.handler(new ChannelInitializer<NioDatagramChannel>() { .handler(new ChannelInitializer<NioDatagramChannel>() {
@Override @Override
protected void initChannel(NioDatagramChannel ch) throws Exception { protected void initChannel(NioDatagramChannel ch)
throws Exception {
ChannelPipeline pipeline = ch.pipeline(); ChannelPipeline pipeline = ch.pipeline();
//0 代表禁用 readerIdleTime 事件的监听 //0 代表禁用 readerIdleTime 事件的监听
//这两个是固定的 //这两个是固定的
@ -267,8 +282,9 @@ public enum UDPClient {
}); });
try { try {
this.conn = this.bootstrap.bind(this.bindPort).sync().channel(); this.conn = this.bootstrap.bind(this.bindPort).sync().channel();
}catch (Exception e){ }
logger.info("AGV UDP Initial Exception : "+e.getMessage()); catch (Exception e) {
logger.info("AGV UDP Initial Exception : " + e.getMessage());
} }
} }

View File

@ -86,38 +86,44 @@ class SameThreadExecutorServiceTest {
verify(task).call(); verify(task).call();
} }
@Test /*
void testVehicle(){ * @Test
//获取车辆对象代码 * void testVehicle() {
KernelServicePortal servicePortal = new KernelServicePortalBuilder(GuestUserCredentials.USER,GuestUserCredentials.PASSWORD).build(); * //获取车辆对象代码
servicePortal.login(GuestUserCredentials.IP, GuestUserCredentials.PORT); * KernelServicePortal servicePortal = new KernelServicePortalBuilder(
// servicePortal.getPlantModelService().up * GuestUserCredentials.USER, GuestUserCredentials.PASSWORD
PlantModelService plantModelService = servicePortal.getPlantModelService(); * ).build();
VehicleService vehicleService = servicePortal.getVehicleService(); * servicePortal.login(GuestUserCredentials.IP, GuestUserCredentials.PORT);
Set<Vehicle> vehicles = vehicleService.fetchObjects(Vehicle.class); * // servicePortal.getPlantModelService().up
for (Vehicle vehicle : vehicles) { * PlantModelService plantModelService = servicePortal.getPlantModelService();
System.out.println("vehicle:"+vehicle); * VehicleService vehicleService = servicePortal.getVehicleService();
} * Set<Vehicle> vehicles = vehicleService.fetchObjects(Vehicle.class);
} * for (Vehicle vehicle : vehicles) {
* System.out.println("vehicle:" + vehicle);
* }
* }
*/
@Test @Test
void testRemoveVehicle(){ void testRemoveVehicle() {
HashMap<String, Object> map = new HashMap<>(); HashMap<String, Object> map = new HashMap<>();
int energy = 54; int energy = 52;
map.put("energy", energy); map.put("energy", energy);
long positionX = -100; long positionX = 3040;
map.put("positionX", positionX); map.put("positionX", positionX);
long positionY = 6510; long positionY = -40;
map.put("positionY", positionY); map.put("positionY", positionY);
double positionAngle = 0.0; double positionAngle = 0.0;
map.put("positionAngle", positionAngle); map.put("positionAngle", positionAngle);
//向车辆对应的通讯适配器发送消息 //向车辆对应的通讯适配器发送消息
KernelServicePortal servicePortal = new KernelServicePortalBuilder(GuestUserCredentials.USER,GuestUserCredentials.PASSWORD).build(); KernelServicePortal servicePortal = new KernelServicePortalBuilder(
GuestUserCredentials.USER, GuestUserCredentials.PASSWORD
).build();
servicePortal.login(GuestUserCredentials.IP, GuestUserCredentials.PORT); servicePortal.login(GuestUserCredentials.IP, GuestUserCredentials.PORT);
VehicleService vehicleService = servicePortal.getVehicleService(); VehicleService vehicleService = servicePortal.getVehicleService();
Vehicle vehicle = vehicleService.fetchObject(Vehicle.class, "2"); Vehicle vehicle = vehicleService.fetchObject(Vehicle.class, "50");
vehicleService.sendCommAdapterMessage(vehicle.getReference(), map); vehicleService.sendCommAdapterMessage(vehicle.getReference(), map);
} }

View File

@ -42,6 +42,7 @@ public class DefaultPropertySuggestions
keySuggestions.add(LoopbackAdapterConstants.AGV_AUTHORIZE_CODE); keySuggestions.add(LoopbackAdapterConstants.AGV_AUTHORIZE_CODE);
keySuggestions.add(LoopbackAdapterConstants.AGV_IP); keySuggestions.add(LoopbackAdapterConstants.AGV_IP);
keySuggestions.add(LoopbackAdapterConstants.AGV_PORT); keySuggestions.add(LoopbackAdapterConstants.AGV_PORT);
keySuggestions.add(LoopbackAdapterConstants.POINT_TYPE);
} }
@Override @Override