[no commit message]
This commit is contained in:
parent
ac7577a430
commit
2503e22b38
@ -17,21 +17,25 @@ package inr.numass.control.msp;
|
|||||||
|
|
||||||
import hep.dataforge.context.Context;
|
import hep.dataforge.context.Context;
|
||||||
import hep.dataforge.context.GlobalContext;
|
import hep.dataforge.context.GlobalContext;
|
||||||
import hep.dataforge.control.measurements.DataDevice;
|
import hep.dataforge.control.devices.SingleMeasurementDevice;
|
||||||
|
import hep.dataforge.control.measurements.AbstractMeasurement;
|
||||||
|
import hep.dataforge.control.measurements.Measurement;
|
||||||
import hep.dataforge.control.ports.PortHandler;
|
import hep.dataforge.control.ports.PortHandler;
|
||||||
import hep.dataforge.control.ports.TcpPortHandler;
|
import hep.dataforge.control.ports.TcpPortHandler;
|
||||||
import hep.dataforge.data.DataFormat;
|
import hep.dataforge.data.DataFormat;
|
||||||
import hep.dataforge.data.DataFormatBuilder;
|
import hep.dataforge.data.DataFormatBuilder;
|
||||||
|
import hep.dataforge.data.DataPoint;
|
||||||
import hep.dataforge.data.MapDataPoint;
|
import hep.dataforge.data.MapDataPoint;
|
||||||
import hep.dataforge.exceptions.ControlException;
|
import hep.dataforge.exceptions.ControlException;
|
||||||
|
import hep.dataforge.exceptions.MeasurementException;
|
||||||
import hep.dataforge.exceptions.PortException;
|
import hep.dataforge.exceptions.PortException;
|
||||||
import hep.dataforge.exceptions.StorageException;
|
import hep.dataforge.exceptions.StorageException;
|
||||||
import hep.dataforge.meta.Meta;
|
import hep.dataforge.meta.Meta;
|
||||||
import hep.dataforge.storage.api.PointLoader;
|
import hep.dataforge.storage.api.PointLoader;
|
||||||
import hep.dataforge.storage.api.Storage;
|
import hep.dataforge.storage.api.Storage;
|
||||||
import hep.dataforge.storage.commons.LoaderFactory;
|
import hep.dataforge.storage.commons.LoaderFactory;
|
||||||
import hep.dataforge.storage.commons.StoragePlugin;
|
|
||||||
import hep.dataforge.storage.loaders.ChainPointLoader;
|
import hep.dataforge.storage.loaders.ChainPointLoader;
|
||||||
|
import hep.dataforge.values.Value;
|
||||||
import java.time.Instant;
|
import java.time.Instant;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.LinkedHashMap;
|
import java.util.LinkedHashMap;
|
||||||
@ -39,13 +43,14 @@ import java.util.List;
|
|||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Scanner;
|
import java.util.Scanner;
|
||||||
import java.util.concurrent.ConcurrentSkipListMap;
|
import java.util.concurrent.ConcurrentSkipListMap;
|
||||||
import org.slf4j.LoggerFactory;
|
import java.util.logging.Level;
|
||||||
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @author Alexander Nozik
|
* @author Alexander Nozik
|
||||||
*/
|
*/
|
||||||
public class MspDevice extends DataDevice implements PortHandler.PortController {
|
public class MspDevice extends SingleMeasurementDevice {
|
||||||
|
|
||||||
// private static final String PEAK_SET_PATH = "peakJump.peak";
|
// private static final String PEAK_SET_PATH = "peakJump.peak";
|
||||||
private static final int TIMEOUT = 200;
|
private static final int TIMEOUT = 200;
|
||||||
@ -69,9 +74,6 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
private boolean isFilamentOn = false;
|
private boolean isFilamentOn = false;
|
||||||
// private boolean isScanning = false;
|
// private boolean isScanning = false;
|
||||||
|
|
||||||
private final Map<Integer, Double> measurement = new ConcurrentSkipListMap<>();
|
|
||||||
private Map<Integer, String> peakMap;
|
|
||||||
|
|
||||||
public MspDevice(String name, Meta annotation) {
|
public MspDevice(String name, Meta annotation) {
|
||||||
super(name, GlobalContext.instance(), annotation);
|
super(name, GlobalContext.instance(), annotation);
|
||||||
}
|
}
|
||||||
@ -90,46 +92,76 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
handler.setDelimeter("\r\r");
|
handler.setDelimeter("\r\r");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Measurement createMeasurement(Meta meta) throws ControlException {
|
||||||
|
switch (meta.getString("type", "peakJump")) {
|
||||||
|
case "peakJump":
|
||||||
|
return new PeakJumpMeasurement(meta);
|
||||||
|
default:
|
||||||
|
throw new ControlException("Unknown measurement type");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Object calculateState(String stateName) throws ControlException {
|
||||||
|
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String type() {
|
||||||
|
return "MKS E-Vision";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean applyState(String stateName, Value stateValue) throws ControlException {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Startup MSP: get available sensors, select sensor and control.
|
* Startup MSP: get available sensors, select sensor and control.
|
||||||
*
|
*
|
||||||
* @param measurement
|
* @param measurement
|
||||||
* @throws hep.dataforge.exceptions.PortException
|
* @throws hep.dataforge.exceptions.PortException
|
||||||
*/
|
*/
|
||||||
@Override
|
public void setConnected(boolean connected) throws ControlException {
|
||||||
public void doStart(Meta measurement) throws ControlException {
|
if (getState("connection").booleanValue() != connected) {
|
||||||
if (!isControlled) {
|
// if (handler.isLocked()) {
|
||||||
if (handler.isLocked()) {
|
// LoggerFactory.getLogger(getClass()).error("Trying to init MSP controller on locked port. Breaking the lock.");
|
||||||
LoggerFactory.getLogger(getClass()).error("Trying to init MSP controller on locked port. Breaking the lock.");
|
// handler.breakHold();
|
||||||
handler.breakHold();
|
// }
|
||||||
}
|
// handler.holdBy(this);
|
||||||
handler.holdBy(this);
|
if (connected) {
|
||||||
MspResponse response = sendAndWait("Sensors");
|
MspResponse response = sendAndWait("Sensors");
|
||||||
if (response.isOK()) {
|
if (response.isOK()) {
|
||||||
this.sensorName = response.get(2, 1);
|
this.sensorName = response.get(2, 1);
|
||||||
} else {
|
} else {
|
||||||
error(response.errorDescription(), null);
|
evaluateError(response.errorDescription(), null);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
//PENDING определеить в конфиге номер прибора
|
//PENDING определеить в конфиге номер прибора
|
||||||
|
|
||||||
response = sendAndWait("Select", sensorName);
|
response = sendAndWait("Select", sensorName);
|
||||||
if (response.isOK()) {
|
if (response.isOK()) {
|
||||||
this.isSelected = true;
|
this.isSelected = true;
|
||||||
|
} else {
|
||||||
|
evaluateError(response.errorDescription(), null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
response = sendAndWait("Control", "inr.numass.msp", "1.0");
|
||||||
|
if (response.isOK()) {
|
||||||
|
this.isControlled = true;
|
||||||
|
} else {
|
||||||
|
evaluateError(response.errorDescription(), null);
|
||||||
|
}
|
||||||
|
updateState("", TIMEOUT);
|
||||||
} else {
|
} else {
|
||||||
error(response.errorDescription(), null);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
response = sendAndWait("Control", "inr.numass.msp", "1.0");
|
|
||||||
if (response.isOK()) {
|
|
||||||
this.isControlled = true;
|
|
||||||
} else {
|
|
||||||
error(response.errorDescription(), null);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
createPeakJumpMeasurement(buildMeasurementLaminate(measurement).getNode("peakJump"));
|
// createPeakJumpMeasurement(buildMeasurementLaminate(measurement).getNode("peakJump"));
|
||||||
this.peakJumpLoader = getPeakJumpLoader(measurement);
|
// this.peakJumpLoader = getPeakJumpLoader(measurement);
|
||||||
}
|
}
|
||||||
|
|
||||||
// public void setStorageConfig(Meta storageConfig, List<String> peaks) throws StorageException {
|
// public void setStorageConfig(Meta storageConfig, List<String> peaks) throws StorageException {
|
||||||
@ -138,7 +170,6 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
//
|
//
|
||||||
// this.peakJumpLoader = LoaderFactory.buildPointLoder(storage, "msp" + suffix, "", "timestamp", DataFormat.forNames(10, peaks));
|
// this.peakJumpLoader = LoaderFactory.buildPointLoder(storage, "msp" + suffix, "", "timestamp", DataFormat.forNames(10, peaks));
|
||||||
// }
|
// }
|
||||||
|
|
||||||
private PointLoader getPeakJumpLoader(Meta measurement) {
|
private PointLoader getPeakJumpLoader(Meta measurement) {
|
||||||
if (peakJumpLoader == null) {
|
if (peakJumpLoader == null) {
|
||||||
try {
|
try {
|
||||||
@ -158,14 +189,12 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
DataFormat format = builder.build();
|
DataFormat format = builder.build();
|
||||||
|
|
||||||
//TODO Переделать!!!
|
//TODO Переделать!!!
|
||||||
String run = meta().getString("numass.run","");
|
String run = meta().getString("numass.run", "");
|
||||||
|
|
||||||
String suffix = Integer.toString((int) Instant.now().toEpochMilli());
|
String suffix = Integer.toString((int) Instant.now().toEpochMilli());
|
||||||
peakJumpLoader = LoaderFactory
|
peakJumpLoader = LoaderFactory
|
||||||
.buildPointLoder(primaryStorage, "msp" + suffix, run, "timestamp", format);
|
.buildPointLoder(primaryStorage, "msp" + suffix, run, "timestamp", format);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Storage secondaryStorage = getSecondaryStorage(measurement);
|
Storage secondaryStorage = getSecondaryStorage(measurement);
|
||||||
if (secondaryStorage != null) {
|
if (secondaryStorage != null) {
|
||||||
@ -190,6 +219,13 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
this.mspListener = listener;
|
this.mspListener = listener;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send request to the msp
|
||||||
|
*
|
||||||
|
* @param command
|
||||||
|
* @param parameters
|
||||||
|
* @throws PortException
|
||||||
|
*/
|
||||||
private void send(String command, Object... parameters) throws PortException {
|
private void send(String command, Object... parameters) throws PortException {
|
||||||
String request = buildCommand(command, parameters);
|
String request = buildCommand(command, parameters);
|
||||||
if (mspListener != null) {
|
if (mspListener != null) {
|
||||||
@ -198,6 +234,13 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
handler.send(request);
|
handler.send(request);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A helper method to build msp command string
|
||||||
|
*
|
||||||
|
* @param command
|
||||||
|
* @param parameters
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
private String buildCommand(String command, Object... parameters) {
|
private String buildCommand(String command, Object... parameters) {
|
||||||
StringBuilder builder = new StringBuilder(command);
|
StringBuilder builder = new StringBuilder(command);
|
||||||
for (Object par : parameters) {
|
for (Object par : parameters) {
|
||||||
@ -208,7 +251,7 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Send specific command and wait for its results (the result must begin
|
* Send specific command and wait for its results (the onResult must begin
|
||||||
* with command name)
|
* with command name)
|
||||||
*
|
*
|
||||||
* @param commandName
|
* @param commandName
|
||||||
@ -231,72 +274,6 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
return new MspResponse(response);
|
return new MspResponse(response);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public synchronized void accept(String message) {
|
|
||||||
if (mspListener != null) {
|
|
||||||
mspListener.acceptMessage(message.trim());
|
|
||||||
}
|
|
||||||
|
|
||||||
MspResponse response = new MspResponse(message);
|
|
||||||
switch (response.getCommandName()) {
|
|
||||||
// all possible async messages
|
|
||||||
case "FilamentStatus":
|
|
||||||
String status = response.get(0, 2);
|
|
||||||
isFilamentOn = status.equals("ON");
|
|
||||||
if (this.mspListener != null) {
|
|
||||||
this.mspListener.acceptFillamentStateChange(status);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "MassReading":
|
|
||||||
double mass = Double.parseDouble(response.get(0, 1));
|
|
||||||
double value = Double.parseDouble(response.get(0, 2));
|
|
||||||
this.measurement.put((int) Math.floor(mass + 0.5), value);
|
|
||||||
break;
|
|
||||||
case "StartingScan":
|
|
||||||
if (this.mspListener != null && !measurement.isEmpty() && isFilamentOn) {
|
|
||||||
|
|
||||||
if (peakMap == null) {
|
|
||||||
throw new IllegalStateException("Peal map is not initialized");
|
|
||||||
}
|
|
||||||
|
|
||||||
mspListener.acceptMeasurement(measurement);
|
|
||||||
|
|
||||||
Instant time = Instant.now();
|
|
||||||
|
|
||||||
MapDataPoint point = new MapDataPoint();
|
|
||||||
point.putValue("timestamp", time);
|
|
||||||
|
|
||||||
for (Map.Entry<Integer, Double> entry : measurement.entrySet()) {
|
|
||||||
double val = entry.getValue();
|
|
||||||
point.putValue(peakMap.get(entry.getKey()), val);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (peakJumpLoader != null) {
|
|
||||||
try {
|
|
||||||
peakJumpLoader.push(point);
|
|
||||||
} catch (StorageException ex) {
|
|
||||||
getLogger().error("Push to repo failed", ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
measurement.clear();
|
|
||||||
|
|
||||||
int numScans = Integer.parseInt(response.get(0, 3));
|
|
||||||
|
|
||||||
if (numScans == 0) {
|
|
||||||
try {
|
|
||||||
send("ScanResume", 2);
|
|
||||||
//FIXME обработать ошибку связи
|
|
||||||
} catch (PortException ex) {
|
|
||||||
error(null, ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isSelected() {
|
public boolean isSelected() {
|
||||||
return isSelected;
|
return isSelected;
|
||||||
}
|
}
|
||||||
@ -328,67 +305,65 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
// /**
|
||||||
* Create measurement with parameters and return its name
|
// * Create measurement with parameters and return its name
|
||||||
*
|
// *
|
||||||
* @param an
|
// * @param an
|
||||||
* @return
|
// * @return
|
||||||
* @throws hep.dataforge.exceptions.PortException
|
// * @throws hep.dataforge.exceptions.PortException
|
||||||
*/
|
// */
|
||||||
private void createPeakJumpMeasurement(Meta an) throws ControlException {
|
// private void createPeakJumpMeasurement(Meta an) throws ControlException {
|
||||||
String name = "peakJump";//an.getString("measurementNAmname", "default");
|
// String name = "peakJump";//an.getString("measurementNAmname", "default");
|
||||||
String filterMode = an.getString("filterMode", "PeakAverage");
|
// String filterMode = an.getString("filterMode", "PeakAverage");
|
||||||
int accuracy = an.getInt("accuracy", 5);
|
// int accuracy = an.getInt("accuracy", 5);
|
||||||
//PENDING вставить остальные параметры?
|
// //PENDING вставить остальные параметры?
|
||||||
sendAndWait("MeasurementRemove", name);
|
// sendAndWait("MeasurementRemove", name);
|
||||||
if (sendAndWait("AddPeakJump", name, filterMode, accuracy, 0, 0, 0).isOK()) {
|
// if (sendAndWait("AddPeakJump", name, filterMode, accuracy, 0, 0, 0).isOK()) {
|
||||||
peakMap = new LinkedHashMap<>();
|
// peakMap = new LinkedHashMap<>();
|
||||||
for (Meta peak : an.getNodes("peak")) {
|
// for (Meta peak : an.getNodes("peak")) {
|
||||||
peakMap.put(peak.getInt("mass"), peak.getString("name", peak.getString("mass")));
|
// peakMap.put(peak.getInt("mass"), peak.getString("name", peak.getString("mass")));
|
||||||
if (!sendAndWait("MeasurementAddMass", peak.getString("mass")).isOK()) {
|
// if (!sendAndWait("MeasurementAddMass", peak.getString("mass")).isOK()) {
|
||||||
throw new ControlException("Can't add mass to measurement measurement for msp");
|
// throw new ControlException("Can't add mass to measurement measurement for msp");
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
} else {
|
// } else {
|
||||||
throw new ControlException("Can't create measurement for msp");
|
// throw new ControlException("Can't create measurement for msp");
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
public boolean startPeakJumpMeasurement() throws PortException {
|
|
||||||
if (!isFilamentOn()) {
|
|
||||||
error("Can't start measurement. Filament is not turned on.", null);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!sendAndWait("ScanAdd", "peakJump").isOK()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return sendAndWait("ScanStart", 2).isOK();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
public boolean stopMeasurement() throws PortException {
|
|
||||||
return sendAndWait("ScanStop").isOK();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void doStop() throws PortException {
|
public void shutdown() throws ControlException {
|
||||||
stopMeasurement();
|
super.shutdown();
|
||||||
|
super.stopMeasurement(true);
|
||||||
setFileamentOn(false);
|
setFileamentOn(false);
|
||||||
sendAndWait("Release");
|
sendAndWait("Release");
|
||||||
handler.unholdBy(this);
|
|
||||||
handler.close();
|
handler.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
/**
|
||||||
public void error(String errorMessage, Throwable error) {
|
* Evaluate general async messages
|
||||||
|
*
|
||||||
|
* @param response
|
||||||
|
*/
|
||||||
|
private void evaluateResponse(MspResponse response) {
|
||||||
|
switch (response.getCommandName()) {
|
||||||
|
// all possible async messages
|
||||||
|
case "FilamentStatus":
|
||||||
|
String status = response.get(0, 2);
|
||||||
|
isFilamentOn = status.equals("ON");
|
||||||
|
if (mspListener != null) {
|
||||||
|
mspListener.acceptFillamentStateChange(status);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void evaluateError(String errorMessage, Throwable error) {
|
||||||
if (mspListener != null) {
|
if (mspListener != null) {
|
||||||
mspListener.error(errorMessage, error);
|
mspListener.error(errorMessage, error);
|
||||||
|
} else if (error != null) {
|
||||||
|
throw new RuntimeException(error);
|
||||||
} else {
|
} else {
|
||||||
if (error != null) {
|
throw new RuntimeException(errorMessage);
|
||||||
throw new RuntimeException(error);
|
|
||||||
} else {
|
|
||||||
throw new RuntimeException(errorMessage);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -443,4 +418,138 @@ public class MspDevice extends DataDevice implements PortHandler.PortController
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private class PeakJumpMeasurement extends AbstractMeasurement<DataPoint> implements PortHandler.PortController {
|
||||||
|
|
||||||
|
private final Map<Integer, Double> measurement = new ConcurrentSkipListMap<>();
|
||||||
|
private Map<Integer, String> peakMap;
|
||||||
|
private final Meta meta;
|
||||||
|
|
||||||
|
public PeakJumpMeasurement(Meta meta) {
|
||||||
|
this.meta = meta;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void start() {
|
||||||
|
try {
|
||||||
|
//Take control of port
|
||||||
|
handler.holdBy(this);
|
||||||
|
} catch (PortException ex) {
|
||||||
|
Logger.getLogger(MspDevice.class.getName()).log(Level.SEVERE, null, ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
String name = "peakJump";//an.getString("measurementNAmname", "default");
|
||||||
|
String filterMode = meta.getString("filterMode", "PeakAverage");
|
||||||
|
int accuracy = meta.getInt("accuracy", 5);
|
||||||
|
//PENDING вставить остальные параметры?
|
||||||
|
sendAndWait("MeasurementRemove", name);
|
||||||
|
if (sendAndWait("AddPeakJump", name, filterMode, accuracy, 0, 0, 0).isOK()) {
|
||||||
|
peakMap = new LinkedHashMap<>();
|
||||||
|
for (Meta peak : meta.getNodes("peak")) {
|
||||||
|
peakMap.put(peak.getInt("mass"), peak.getString("name", peak.getString("mass")));
|
||||||
|
if (!sendAndWait("MeasurementAddMass", peak.getString("mass")).isOK()) {
|
||||||
|
throw new ControlException("Can't add mass to measurement measurement for msp");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
throw new ControlException("Can't create measurement for msp");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isFilamentOn()) {
|
||||||
|
this.error("Can't start measurement. Filament is not turned on.", null);
|
||||||
|
}
|
||||||
|
if (!sendAndWait("ScanAdd", "peakJump").isOK()) {
|
||||||
|
this.error("Failed to add scan", null);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!sendAndWait("ScanStart", 2).isOK()) {
|
||||||
|
this.error("Failed to start scan", null);
|
||||||
|
}
|
||||||
|
} catch (ControlException ex) {
|
||||||
|
onError(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean stop(boolean force) {
|
||||||
|
handler.unholdBy(this);
|
||||||
|
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void accept(String message) {
|
||||||
|
if (mspListener != null) {
|
||||||
|
mspListener.acceptMessage(message.trim());
|
||||||
|
}
|
||||||
|
|
||||||
|
MspResponse response = new MspResponse(message);
|
||||||
|
|
||||||
|
//Evaluating device state change
|
||||||
|
evaluateResponse(response);
|
||||||
|
//Evaluating measurement information
|
||||||
|
switch (response.getCommandName()) {
|
||||||
|
case "MassReading":
|
||||||
|
double mass = Double.parseDouble(response.get(0, 1));
|
||||||
|
double value = Double.parseDouble(response.get(0, 2));
|
||||||
|
measurement.put((int) Math.floor(mass + 0.5), value);
|
||||||
|
break;
|
||||||
|
case "StartingScan":
|
||||||
|
if (mspListener != null && !measurement.isEmpty() && isFilamentOn) {
|
||||||
|
|
||||||
|
if (peakMap == null) {
|
||||||
|
throw new IllegalStateException("Peal map is not initialized");
|
||||||
|
}
|
||||||
|
|
||||||
|
mspListener.acceptScan(measurement);
|
||||||
|
|
||||||
|
Instant time = Instant.now();
|
||||||
|
|
||||||
|
MapDataPoint point = new MapDataPoint();
|
||||||
|
point.putValue("timestamp", time);
|
||||||
|
|
||||||
|
measurement.entrySet().stream().forEach((entry) -> {
|
||||||
|
double val = entry.getValue();
|
||||||
|
point.putValue(peakMap.get(entry.getKey()), val);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (peakJumpLoader != null) {
|
||||||
|
try {
|
||||||
|
peakJumpLoader.push(point);
|
||||||
|
} catch (StorageException ex) {
|
||||||
|
getLogger().error("Push to repo failed", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
measurement.clear();
|
||||||
|
|
||||||
|
int numScans = Integer.parseInt(response.get(0, 3));
|
||||||
|
|
||||||
|
if (numScans == 0) {
|
||||||
|
try {
|
||||||
|
send("ScanResume", 2);
|
||||||
|
//FIXME обработать ошибку связи
|
||||||
|
} catch (PortException ex) {
|
||||||
|
error(null, ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean stopMeasurement() throws PortException {
|
||||||
|
return sendAndWait("ScanStop").isOK();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void error(String errorMessage, Throwable error) {
|
||||||
|
if (error == null) {
|
||||||
|
onError(new MeasurementException(errorMessage));
|
||||||
|
} else {
|
||||||
|
onError(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -22,10 +22,14 @@ import java.util.Map;
|
|||||||
* @author darksnake
|
* @author darksnake
|
||||||
*/
|
*/
|
||||||
public interface MspListener {
|
public interface MspListener {
|
||||||
|
|
||||||
void error(String errorMessage, Throwable error);
|
void error(String errorMessage, Throwable error);
|
||||||
void acceptMeasurement(Map<Integer, Double> point);
|
|
||||||
|
void acceptScan(Map<Integer, Double> point);
|
||||||
|
|
||||||
void acceptMessage(String message);
|
void acceptMessage(String message);
|
||||||
void acceptRequest(String message);
|
void acceptRequest(String message);
|
||||||
|
|
||||||
default void acceptFillamentStateChange(String fillamentState){
|
default void acceptFillamentStateChange(String fillamentState){
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -155,7 +155,7 @@ public class MspViewController implements Initializable, MspListener {
|
|||||||
try {
|
try {
|
||||||
getDevice().setListener(this);
|
getDevice().setListener(this);
|
||||||
getDevice().init();
|
getDevice().init();
|
||||||
getDevice().start();
|
getDevice().startMeasurement("peakJump");
|
||||||
} catch (ControlException ex) {
|
} catch (ControlException ex) {
|
||||||
showError(String.format("Can't connect to %s:%d. The port is either busy or not the MKS mass-spectrometer port",
|
showError(String.format("Can't connect to %s:%d. The port is either busy or not the MKS mass-spectrometer port",
|
||||||
config.getString("connection.ip", "127.0.0.1"),
|
config.getString("connection.ip", "127.0.0.1"),
|
||||||
@ -224,7 +224,7 @@ public class MspViewController implements Initializable, MspListener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void acceptMeasurement(Map<Integer, Double> measurement) {
|
public void acceptScan(Map<Integer, Double> measurement) {
|
||||||
MapDataPoint point = new MapDataPoint();
|
MapDataPoint point = new MapDataPoint();
|
||||||
for (Map.Entry<Integer, Double> entry : measurement.entrySet()) {
|
for (Map.Entry<Integer, Double> entry : measurement.entrySet()) {
|
||||||
Double val = entry.getValue();
|
Double val = entry.getValue();
|
||||||
@ -294,11 +294,11 @@ public class MspViewController implements Initializable, MspListener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@FXML
|
@FXML
|
||||||
private void onPlotToggle(ActionEvent event) throws PortException {
|
private void onPlotToggle(ActionEvent event) throws ControlException {
|
||||||
if (plotButton.isSelected()) {
|
if (plotButton.isSelected()) {
|
||||||
getDevice().startPeakJumpMeasurement();
|
getDevice().startMeasurement("peakJump");
|
||||||
} else {
|
} else {
|
||||||
getDevice().stopMeasurement();
|
getDevice().stopMeasurement(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -316,7 +316,7 @@ public class MspViewController implements Initializable, MspListener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void disconnect() throws IOException, PortException, ControlException {
|
public void disconnect() throws IOException, PortException, ControlException {
|
||||||
getDevice().stop();
|
getDevice().shutdown();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -6,9 +6,9 @@
|
|||||||
package inr.numass.readvac.devices;
|
package inr.numass.readvac.devices;
|
||||||
|
|
||||||
import hep.dataforge.context.Context;
|
import hep.dataforge.context.Context;
|
||||||
|
import hep.dataforge.control.measurements.SimpletMeasurement;
|
||||||
import hep.dataforge.control.measurements.Measurement;
|
import hep.dataforge.control.measurements.Measurement;
|
||||||
import hep.dataforge.control.measurements.Sensor;
|
import hep.dataforge.control.measurements.Sensor;
|
||||||
import hep.dataforge.control.measurements.SimpleMeasurement;
|
|
||||||
import hep.dataforge.control.ports.ComPortHandler;
|
import hep.dataforge.control.ports.ComPortHandler;
|
||||||
import hep.dataforge.control.ports.PortHandler;
|
import hep.dataforge.control.ports.PortHandler;
|
||||||
import hep.dataforge.description.ValueDef;
|
import hep.dataforge.description.ValueDef;
|
||||||
@ -30,6 +30,10 @@ public class CM32Device extends Sensor<Double> {
|
|||||||
super(name, context, meta);
|
super(name, context, meta);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void setHandler(PortHandler handler){
|
||||||
|
this.handler = handler;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the handler
|
* @return the handler
|
||||||
*/
|
*/
|
||||||
@ -81,29 +85,29 @@ public class CM32Device extends Sensor<Double> {
|
|||||||
return meta().getInt("timeout", 400);
|
return meta().getInt("timeout", 400);
|
||||||
}
|
}
|
||||||
|
|
||||||
private class CMVacMeasurement extends SimpleMeasurement<Double> {
|
private class CMVacMeasurement extends SimpletMeasurement<Double> {
|
||||||
|
|
||||||
private static final String CM32_QUERY = "MES R PM 1\r\n";
|
private static final String CM32_QUERY = "MES R PM 1\r\n";
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Double doMeasurement() throws Exception {
|
protected synchronized Double doMeasure() throws Exception {
|
||||||
|
|
||||||
String answer = handler.sendAndWait(CM32_QUERY, timeout());
|
String answer = handler.sendAndWait(CM32_QUERY, timeout());
|
||||||
|
|
||||||
if (answer.isEmpty()) {
|
if (answer.isEmpty()) {
|
||||||
this.progressUpdate("No signal");
|
this.onProgressUpdate("No signal");
|
||||||
updateState("connection", false);
|
updateState("connection", false);
|
||||||
return null;
|
return null;
|
||||||
} else if (answer.indexOf("PM1:mbar") < -1) {
|
} else if (answer.indexOf("PM1:mbar") < -1) {
|
||||||
this.progressUpdate("Wrong answer: " + answer);
|
this.onProgressUpdate("Wrong answer: " + answer);
|
||||||
updateState("connection", false);
|
updateState("connection", false);
|
||||||
return null;
|
return null;
|
||||||
} else if (answer.substring(14, 17).equals("OFF")) {
|
} else if (answer.substring(14, 17).equals("OFF")) {
|
||||||
this.progressUpdate("Off");
|
this.onProgressUpdate("Off");
|
||||||
updateState("connection", true);
|
updateState("connection", true);
|
||||||
return null;
|
return null;
|
||||||
} else {
|
} else {
|
||||||
this.progressUpdate("OK");
|
this.onProgressUpdate("OK");
|
||||||
updateState("connection", true);
|
updateState("connection", true);
|
||||||
return Double.parseDouble(answer.substring(14, 17) + answer.substring(19, 23));
|
return Double.parseDouble(answer.substring(14, 17) + answer.substring(19, 23));
|
||||||
}
|
}
|
||||||
|
@ -6,9 +6,9 @@
|
|||||||
package inr.numass.readvac.devices;
|
package inr.numass.readvac.devices;
|
||||||
|
|
||||||
import hep.dataforge.context.Context;
|
import hep.dataforge.context.Context;
|
||||||
|
import hep.dataforge.control.measurements.SimpletMeasurement;
|
||||||
import hep.dataforge.control.measurements.Measurement;
|
import hep.dataforge.control.measurements.Measurement;
|
||||||
import hep.dataforge.control.measurements.Sensor;
|
import hep.dataforge.control.measurements.Sensor;
|
||||||
import hep.dataforge.control.measurements.SimpleMeasurement;
|
|
||||||
import hep.dataforge.control.ports.ComPortHandler;
|
import hep.dataforge.control.ports.ComPortHandler;
|
||||||
import hep.dataforge.control.ports.PortHandler;
|
import hep.dataforge.control.ports.PortHandler;
|
||||||
import hep.dataforge.description.ValueDef;
|
import hep.dataforge.description.ValueDef;
|
||||||
@ -38,6 +38,10 @@ public class MKSVacDevice extends Sensor<Double> {
|
|||||||
super(name, context, meta);
|
super(name, context, meta);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void setHandler(PortHandler handler){
|
||||||
|
this.handler = handler;
|
||||||
|
}
|
||||||
|
|
||||||
private String talk(String requestContent) throws ControlException {
|
private String talk(String requestContent) throws ControlException {
|
||||||
String answer = getHandler().sendAndWait(String.format("@%s%s;FF", getDeviceAddress(), requestContent), timeout());
|
String answer = getHandler().sendAndWait(String.format("@%s%s;FF", getDeviceAddress(), requestContent), timeout());
|
||||||
|
|
||||||
@ -171,23 +175,23 @@ public class MKSVacDevice extends Sensor<Double> {
|
|||||||
return handler;
|
return handler;
|
||||||
}
|
}
|
||||||
|
|
||||||
private class MKSVacMeasurement extends SimpleMeasurement<Double> {
|
private class MKSVacMeasurement extends SimpletMeasurement<Double> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Double doMeasurement() throws Exception {
|
protected synchronized Double doMeasure() throws Exception {
|
||||||
String answer = talk("PR" + getChannel() + "?");
|
String answer = talk("PR" + getChannel() + "?");
|
||||||
if (answer == null || answer.isEmpty()) {
|
if (answer == null || answer.isEmpty()) {
|
||||||
invalidateState("connection");
|
invalidateState("connection");
|
||||||
this.progressUpdate("No connection");
|
this.onProgressUpdate("No connection");
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
double res = Double.parseDouble(answer);
|
double res = Double.parseDouble(answer);
|
||||||
if (res <= 0) {
|
if (res <= 0) {
|
||||||
this.progressUpdate("Non positive");
|
this.onProgressUpdate("Non positive");
|
||||||
invalidateState("power");
|
invalidateState("power");
|
||||||
return null;
|
return null;
|
||||||
} else {
|
} else {
|
||||||
this.progressUpdate("OK");
|
this.onProgressUpdate("OK");
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,9 +6,9 @@
|
|||||||
package inr.numass.readvac.devices;
|
package inr.numass.readvac.devices;
|
||||||
|
|
||||||
import hep.dataforge.context.Context;
|
import hep.dataforge.context.Context;
|
||||||
|
import hep.dataforge.control.measurements.SimpletMeasurement;
|
||||||
import hep.dataforge.control.measurements.Measurement;
|
import hep.dataforge.control.measurements.Measurement;
|
||||||
import hep.dataforge.control.measurements.Sensor;
|
import hep.dataforge.control.measurements.Sensor;
|
||||||
import hep.dataforge.control.measurements.SimpleMeasurement;
|
|
||||||
import hep.dataforge.control.ports.ComPortHandler;
|
import hep.dataforge.control.ports.ComPortHandler;
|
||||||
import hep.dataforge.control.ports.PortHandler;
|
import hep.dataforge.control.ports.PortHandler;
|
||||||
import hep.dataforge.description.ValueDef;
|
import hep.dataforge.description.ValueDef;
|
||||||
@ -34,6 +34,10 @@ public class VITVacDevice extends Sensor<Double> {
|
|||||||
super(name, context, meta);
|
super(name, context, meta);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void setHandler(PortHandler handler) {
|
||||||
|
this.handler = handler;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the handler
|
* @return the handler
|
||||||
*/
|
*/
|
||||||
@ -85,17 +89,17 @@ public class VITVacDevice extends Sensor<Double> {
|
|||||||
return meta().getInt("timeout", 400);
|
return meta().getInt("timeout", 400);
|
||||||
}
|
}
|
||||||
|
|
||||||
private class CMVacMeasurement extends SimpleMeasurement<Double> {
|
private class CMVacMeasurement extends SimpletMeasurement<Double> {
|
||||||
|
|
||||||
private static final String VIT_QUERY = ":010300000002FA\r\n";
|
private static final String VIT_QUERY = ":010300000002FA\r\n";
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Double doMeasurement() throws Exception {
|
protected synchronized Double doMeasure() throws Exception {
|
||||||
|
|
||||||
String answer = handler.sendAndWait(VIT_QUERY, timeout());
|
String answer = handler.sendAndWait(VIT_QUERY, timeout());
|
||||||
|
|
||||||
if (answer.isEmpty()) {
|
if (answer.isEmpty()) {
|
||||||
this.progressUpdate("No signal");
|
this.onProgressUpdate("No signal");
|
||||||
updateState("connection", false);
|
updateState("connection", false);
|
||||||
return null;
|
return null;
|
||||||
} else {
|
} else {
|
||||||
@ -109,11 +113,11 @@ public class VITVacDevice extends Sensor<Double> {
|
|||||||
}
|
}
|
||||||
BigDecimal res = BigDecimal.valueOf(base * Math.pow(10, exp));
|
BigDecimal res = BigDecimal.valueOf(base * Math.pow(10, exp));
|
||||||
res = res.setScale(4, RoundingMode.CEILING);
|
res = res.setScale(4, RoundingMode.CEILING);
|
||||||
this.progressUpdate("OK");
|
this.onProgressUpdate("OK");
|
||||||
updateState("connection", true);
|
updateState("connection", true);
|
||||||
return res.doubleValue();
|
return res.doubleValue();
|
||||||
} else {
|
} else {
|
||||||
this.progressUpdate("Wrong answer: " + answer);
|
this.onProgressUpdate("Wrong answer: " + answer);
|
||||||
updateState("connection", false);
|
updateState("connection", false);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -13,11 +13,15 @@ import hep.dataforge.control.measurements.Measurement;
|
|||||||
import hep.dataforge.control.measurements.Sensor;
|
import hep.dataforge.control.measurements.Sensor;
|
||||||
import hep.dataforge.data.DataPoint;
|
import hep.dataforge.data.DataPoint;
|
||||||
import hep.dataforge.exceptions.ControlException;
|
import hep.dataforge.exceptions.ControlException;
|
||||||
|
import hep.dataforge.exceptions.MeasurementException;
|
||||||
import hep.dataforge.meta.Meta;
|
import hep.dataforge.meta.Meta;
|
||||||
import java.time.Instant;
|
import hep.dataforge.values.Value;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import javafx.util.Pair;
|
import java.util.concurrent.Executors;
|
||||||
|
import java.util.concurrent.ScheduledExecutorService;
|
||||||
|
import java.util.concurrent.ScheduledFuture;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
@ -44,7 +48,7 @@ public class VacCollectorDevice extends Sensor<DataPoint> {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Measurement<DataPoint> createMeasurement() {
|
protected Measurement<DataPoint> createMeasurement() {
|
||||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
return new VacuumMeasurement();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -54,28 +58,38 @@ public class VacCollectorDevice extends Sensor<DataPoint> {
|
|||||||
|
|
||||||
private class VacuumMeasurement extends AbstractMeasurement<DataPoint> {
|
private class VacuumMeasurement extends AbstractMeasurement<DataPoint> {
|
||||||
|
|
||||||
ValueCollector collector = new PointCollector(this::result, sensorMap.keySet());
|
private final ValueCollector collector = new PointCollector(this::onResult, sensorMap.keySet());
|
||||||
|
private final ScheduledExecutorService executor = Executors.newSingleThreadScheduledExecutor();
|
||||||
@Override
|
private ScheduledFuture<?> currentTask;
|
||||||
protected Pair<DataPoint, Instant> doGet() throws Exception {
|
|
||||||
sensorMap.values().stream().parallel().forEach(action);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean isFinished() {
|
|
||||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void start() {
|
public void start() {
|
||||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
currentTask = executor.scheduleWithFixedDelay(() -> {
|
||||||
|
sensorMap.entrySet().stream().parallel().forEach((entry) -> {
|
||||||
|
try {
|
||||||
|
collector.put(entry.getKey(), entry.getValue().getMeasurement().getResult());
|
||||||
|
} catch (MeasurementException ex) {
|
||||||
|
onError(ex);
|
||||||
|
collector.put(entry.getKey(), Value.NULL);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, 0, getDelay(), TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
private int getDelay() {
|
||||||
|
return meta().getInt("delay", 5000);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean stop(boolean force) {
|
public boolean stop(boolean force) {
|
||||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
boolean isRunning = currentTask != null;
|
||||||
|
if (isRunning) {
|
||||||
|
currentTask.cancel(force);
|
||||||
|
isFinished = true;
|
||||||
|
currentTask = null;
|
||||||
|
}
|
||||||
|
return isRunning;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user