org.bidib.jbidibc.usbstickbasis.adapter.UsbStickBasisAdapter Maven / Gradle / Ivy
package org.bidib.jbidibc.usbstickbasis.adapter;
import java.nio.charset.Charset;
import java.util.LinkedList;
import java.util.List;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.regex.Pattern;
import java.util.stream.Collectors;
import javax.swing.SwingUtilities;
import org.apache.commons.lang3.StringUtils;
import org.bidib.jbidibc.debug.DebugInterface;
import org.bidib.jbidibc.debug.DebugMessageListener;
import org.bidib.jbidibc.debug.DebugMessageReceiver;
import org.bidib.jbidibc.debug.DebugReaderFactory;
import org.bidib.jbidibc.messages.ConnectionListener;
import org.bidib.jbidibc.messages.exception.PortNotFoundException;
import org.bidib.jbidibc.messages.exception.PortNotOpenedException;
import org.bidib.jbidibc.messages.exception.ProtocolException;
import org.bidib.jbidibc.messages.helpers.Context;
import org.bidib.jbidibc.messages.message.BidibMessageInterface;
import org.bidib.jbidibc.messages.message.ResponseFactory;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class UsbStickBasisAdapter {
private static final Logger LOGGER = LoggerFactory.getLogger(UsbStickBasisAdapter.class);
private DebugInterface debugReader;
private DebugMessageReceiver messageReceiver;
private DebugMessageListener messageListener;
private final UsbStickBasisModel usbStickBasisModel;
private final UsbStickBasisResponseInterface responseInterface;
private final StringBuilder resultBuffer = new StringBuilder();
private final List receivedLines = new LinkedList<>();
protected final ScheduledExecutorService receiverWorker = Executors.newScheduledThreadPool(1);
private ResponseFactory responseFactory;
private AtomicBoolean receiverWorkerActive = new AtomicBoolean();
public UsbStickBasisAdapter(final UsbStickBasisResponseInterface responseInterface,
final UsbStickBasisModel usbStickBasisModel) {
this.responseInterface = responseInterface;
this.usbStickBasisModel = usbStickBasisModel;
}
public void openConnection(String portName, String serialPortProvider)
throws PortNotFoundException, PortNotOpenedException {
Integer baudRate = usbStickBasisModel.getBaudRate();
if (baudRate == null) {
LOGGER.warn("No baudrate selected!");
throw new IllegalArgumentException("No baudrate selected.");
}
LOGGER.info("Selected portName: {}", portName);
if (debugReader == null) {
LOGGER.info("Create new instance of debug reader.");
responseFactory = new UsbStickBasisResponseFactory(responseInterface);
responseFactory.initialize();
try {
messageListener = new DebugMessageListener() {
@Override
public void debugMessage(final String message) {
LOGGER.info("debug message received size: {}, message: {}", message.length(), message);
responseInterface.addLog(message);
synchronized (resultBuffer) {
LOGGER.trace("++ append");
resultBuffer.append(message);
resultBuffer.notifyAll();
LOGGER.trace("-- append");
}
}
};
messageReceiver = new DebugMessageReceiver();
messageReceiver.addMessageListener(messageListener);
// create the debug reader
LOGGER.info("Selected serial port provider: {}", serialPortProvider);
DebugReaderFactory.SerialImpl serialImpl = null;
switch (serialPortProvider) {
case "scm":
serialImpl = DebugReaderFactory.SerialImpl.SCM;
break;
default:
// serialImpl = DebugReaderFactory.SerialImpl.RXTX;
serialImpl = DebugReaderFactory.SerialImpl.PUREJAVACOMM;
break;
}
debugReader = DebugReaderFactory.getDebugReader(serialImpl, messageReceiver);
LOGGER.info("Created debugReader: {}", debugReader);
}
catch (Exception ex) {
LOGGER.warn("Open debug port failed.", ex);
}
}
try {
debugReader.open(portName, baudRate, new ConnectionListener() {
@Override
public void opened(String port) {
LOGGER.info("Port opened: {}", port);
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
usbStickBasisModel.setConnected(true);
}
});
}
@Override
public void closed(String port) {
LOGGER.info("Port closed: {}", port);
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
usbStickBasisModel.setConnected(false);
}
});
}
@Override
public void status(String messageKey, final Context context) {
// no implementation
}
@Override
public void stall(boolean stall) {
// no implementation
}
}, null);
int delay = 0;
// use executor to process the responses
LOGGER.info("Schedule the receiverWorker.");
receiverWorkerActive.set(true);
receiverWorker.schedule(new Runnable() {
@Override
public void run() {
LOGGER.info("Start process the receive queue.");
while (receiverWorkerActive.get()) {
try {
processReceiveQueue(responseInterface);
}
catch (ProtocolException ex) {
LOGGER.warn("Process received response failed", ex);
}
}
LOGGER.info("Receive queue worker has finished.");
}
}, delay, TimeUnit.MILLISECONDS);
}
catch (PortNotFoundException |
PortNotOpenedException ex) {
LOGGER.warn("Open serial port failed.", ex);
throw ex;
}
catch (
Exception ex) {
LOGGER.warn("Open serial port failed.", ex);
throw new PortNotOpenedException("Open serial port failed.", ex);
}
}
public void closeConnection() {
LOGGER.info("Close the debug connection.");
if (debugReader != null) {
LOGGER.info("Close the debug reader.");
debugReader.close();
receiverWorkerActive.set(false);
usbStickBasisModel.setConnected(false);
try {
synchronized (resultBuffer) {
resultBuffer.notifyAll();
}
}
catch (Exception ex) {
LOGGER.warn("Notify the resultBuffer failed.", ex);
}
messageReceiver.removeMessageListener(messageListener);
messageListener = null;
messageReceiver = null;
debugReader = null;
}
else {
LOGGER.info("debug reader not available.");
}
}
public void clearReceiveBuffer() {
LOGGER.info("Clear the receive buffer.");
synchronized (resultBuffer) {
resultBuffer.setLength(0);
}
}
public String transmit(String sendText) {
if (StringUtils.isEmpty(sendText)) {
LOGGER.info("No data to send!");
return null;
}
LOGGER.info("Send text to debugReader: '{}'", sendText);
String result = null;
if (debugReader != null) {
debugReader.send(sendText, usbStickBasisModel.getLineEnding());
}
return result;
}
protected void processReceiveQueue(final UsbStickBasisResponseInterface responseInterface)
throws ProtocolException {
synchronized (resultBuffer) {
LOGGER.trace("++ processReceiveQueue, resultBuffer.length(): {}", resultBuffer.length());
if (resultBuffer.length() < 1) {
try {
resultBuffer.wait(500);
}
catch (InterruptedException ex) {
LOGGER.warn("Wait for data in result buffer failed.", ex);
}
}
if (resultBuffer.length() > 0) {
int lastLF = resultBuffer.lastIndexOf("\n");
LOGGER.trace("lastLF: {}", lastLF);
if (lastLF > 0) {
// append the lines
List parts =
Pattern
.compile("\n").splitAsStream(resultBuffer.substring(0, lastLF)).map(str -> str.trim())
.filter(str -> !str.isEmpty()).collect(Collectors.toList());
receivedLines.addAll(parts);
String remaining = null;
if (lastLF < resultBuffer.length()) {
remaining = resultBuffer.substring(lastLF).trim();
LOGGER.info("Keep the remaining: '{}'", remaining);
}
// clear the buffer
resultBuffer.setLength(0);
if (StringUtils.isNotBlank(remaining)) {
resultBuffer.append(remaining);
}
}
}
while (!receivedLines.isEmpty()) {
try {
String part = receivedLines.remove(0).trim();
LOGGER.info("Check part: '{}'", part);
final BidibMessageInterface bidibMessage =
responseFactory.create(part.getBytes(Charset.forName("UTF-8")));
LOGGER.info("Created bidibMessage: {}", bidibMessage);
if (bidibMessage != null) {
responseInterface.publishReponse(bidibMessage);
}
}
catch (Exception ex) {
LOGGER.warn("Create bidib message from received data failed.", ex);
}
}
LOGGER.trace("-- processReceiveQueue");
}
// sleep for 10ms
try {
Thread.sleep(10);
}
catch (Exception ex) {
LOGGER.warn("Sleep for 10ms after processing resultBuffer failed.", ex);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy