Improve behavior with synchronous and asynchronous receivers
This commit is contained in:
parent
eec3d782d3
commit
9096229637
@ -1,5 +1,6 @@
|
|||||||
package org.asamk.signal.manager;
|
package org.asamk.signal.manager;
|
||||||
|
|
||||||
|
import org.asamk.signal.manager.api.AlreadyReceivingException;
|
||||||
import org.asamk.signal.manager.api.AttachmentInvalidException;
|
import org.asamk.signal.manager.api.AttachmentInvalidException;
|
||||||
import org.asamk.signal.manager.api.Configuration;
|
import org.asamk.signal.manager.api.Configuration;
|
||||||
import org.asamk.signal.manager.api.Device;
|
import org.asamk.signal.manager.api.Device;
|
||||||
@ -204,7 +205,7 @@ public interface Manager extends Closeable {
|
|||||||
*/
|
*/
|
||||||
public void receiveMessages(
|
public void receiveMessages(
|
||||||
Optional<Duration> timeout, Optional<Integer> maxMessages, ReceiveMessageHandler handler
|
Optional<Duration> timeout, Optional<Integer> maxMessages, ReceiveMessageHandler handler
|
||||||
) throws IOException;
|
) throws IOException, AlreadyReceivingException;
|
||||||
|
|
||||||
void setReceiveConfig(ReceiveConfig receiveConfig);
|
void setReceiveConfig(ReceiveConfig receiveConfig);
|
||||||
|
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
*/
|
*/
|
||||||
package org.asamk.signal.manager;
|
package org.asamk.signal.manager;
|
||||||
|
|
||||||
|
import org.asamk.signal.manager.api.AlreadyReceivingException;
|
||||||
import org.asamk.signal.manager.api.AttachmentInvalidException;
|
import org.asamk.signal.manager.api.AttachmentInvalidException;
|
||||||
import org.asamk.signal.manager.api.Configuration;
|
import org.asamk.signal.manager.api.Configuration;
|
||||||
import org.asamk.signal.manager.api.Device;
|
import org.asamk.signal.manager.api.Device;
|
||||||
@ -25,6 +26,7 @@ import org.asamk.signal.manager.api.InactiveGroupLinkException;
|
|||||||
import org.asamk.signal.manager.api.InvalidDeviceLinkException;
|
import org.asamk.signal.manager.api.InvalidDeviceLinkException;
|
||||||
import org.asamk.signal.manager.api.InvalidStickerException;
|
import org.asamk.signal.manager.api.InvalidStickerException;
|
||||||
import org.asamk.signal.manager.api.Message;
|
import org.asamk.signal.manager.api.Message;
|
||||||
|
import org.asamk.signal.manager.api.MessageEnvelope;
|
||||||
import org.asamk.signal.manager.api.NotPrimaryDeviceException;
|
import org.asamk.signal.manager.api.NotPrimaryDeviceException;
|
||||||
import org.asamk.signal.manager.api.Pair;
|
import org.asamk.signal.manager.api.Pair;
|
||||||
import org.asamk.signal.manager.api.PendingAdminApprovalException;
|
import org.asamk.signal.manager.api.PendingAdminApprovalException;
|
||||||
@ -874,9 +876,6 @@ class ManagerImpl implements Manager {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void addReceiveHandler(final ReceiveMessageHandler handler, final boolean isWeakListener) {
|
public void addReceiveHandler(final ReceiveMessageHandler handler, final boolean isWeakListener) {
|
||||||
if (isReceivingSynchronous) {
|
|
||||||
throw new IllegalStateException("Already receiving message synchronously.");
|
|
||||||
}
|
|
||||||
synchronized (messageHandlers) {
|
synchronized (messageHandlers) {
|
||||||
if (isWeakListener) {
|
if (isWeakListener) {
|
||||||
weakHandlers.add(handler);
|
weakHandlers.add(handler);
|
||||||
@ -890,23 +889,12 @@ class ManagerImpl implements Manager {
|
|||||||
private static final AtomicInteger threadNumber = new AtomicInteger(0);
|
private static final AtomicInteger threadNumber = new AtomicInteger(0);
|
||||||
|
|
||||||
private void startReceiveThreadIfRequired() {
|
private void startReceiveThreadIfRequired() {
|
||||||
if (receiveThread != null) {
|
if (receiveThread != null || isReceivingSynchronous) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
receiveThread = new Thread(() -> {
|
receiveThread = new Thread(() -> {
|
||||||
logger.debug("Starting receiving messages");
|
logger.debug("Starting receiving messages");
|
||||||
context.getReceiveHelper().receiveMessagesContinuously((envelope, e) -> {
|
context.getReceiveHelper().receiveMessagesContinuously(this::passReceivedMessageToHandlers);
|
||||||
synchronized (messageHandlers) {
|
|
||||||
final var handlers = Stream.concat(messageHandlers.stream(), weakHandlers.stream()).toList();
|
|
||||||
handlers.forEach(h -> {
|
|
||||||
try {
|
|
||||||
h.handleMessage(envelope, e);
|
|
||||||
} catch (Throwable ex) {
|
|
||||||
logger.warn("Message handler failed, ignoring", ex);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
logger.debug("Finished receiving messages");
|
logger.debug("Finished receiving messages");
|
||||||
synchronized (messageHandlers) {
|
synchronized (messageHandlers) {
|
||||||
receiveThread = null;
|
receiveThread = null;
|
||||||
@ -923,6 +911,18 @@ class ManagerImpl implements Manager {
|
|||||||
receiveThread.start();
|
receiveThread.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void passReceivedMessageToHandlers(MessageEnvelope envelope, Throwable e) {
|
||||||
|
synchronized (messageHandlers) {
|
||||||
|
Stream.concat(messageHandlers.stream(), weakHandlers.stream()).forEach(h -> {
|
||||||
|
try {
|
||||||
|
h.handleMessage(envelope, e);
|
||||||
|
} catch (Throwable ex) {
|
||||||
|
logger.warn("Message handler failed, ignoring", ex);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void removeReceiveHandler(final ReceiveMessageHandler handler) {
|
public void removeReceiveHandler(final ReceiveMessageHandler handler) {
|
||||||
final Thread thread;
|
final Thread thread;
|
||||||
@ -962,26 +962,34 @@ class ManagerImpl implements Manager {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void receiveMessages(
|
public void receiveMessages(
|
||||||
Optional<Duration> timeout,
|
Optional<Duration> timeout, Optional<Integer> maxMessages, ReceiveMessageHandler handler
|
||||||
Optional<Integer> maxMessages,
|
) throws IOException, AlreadyReceivingException {
|
||||||
ReceiveMessageHandler handler
|
|
||||||
) throws IOException {
|
|
||||||
receiveMessages(timeout.orElse(Duration.ofMinutes(1)), timeout.isPresent(), maxMessages.orElse(null), handler);
|
receiveMessages(timeout.orElse(Duration.ofMinutes(1)), timeout.isPresent(), maxMessages.orElse(null), handler);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void receiveMessages(
|
private void receiveMessages(
|
||||||
Duration timeout, boolean returnOnTimeout, Integer maxMessages, ReceiveMessageHandler handler
|
Duration timeout, boolean returnOnTimeout, Integer maxMessages, ReceiveMessageHandler handler
|
||||||
) throws IOException {
|
) throws IOException, AlreadyReceivingException {
|
||||||
|
synchronized (messageHandlers) {
|
||||||
if (isReceiving()) {
|
if (isReceiving()) {
|
||||||
throw new IllegalStateException("Already receiving message.");
|
throw new AlreadyReceivingException("Already receiving message.");
|
||||||
}
|
}
|
||||||
isReceivingSynchronous = true;
|
isReceivingSynchronous = true;
|
||||||
receiveThread = Thread.currentThread();
|
receiveThread = Thread.currentThread();
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
context.getReceiveHelper().receiveMessages(timeout, returnOnTimeout, maxMessages, handler);
|
context.getReceiveHelper().receiveMessages(timeout, returnOnTimeout, maxMessages, (envelope, e) -> {
|
||||||
|
passReceivedMessageToHandlers(envelope, e);
|
||||||
|
handler.handleMessage(envelope, e);
|
||||||
|
});
|
||||||
} finally {
|
} finally {
|
||||||
|
synchronized (messageHandlers) {
|
||||||
receiveThread = null;
|
receiveThread = null;
|
||||||
isReceivingSynchronous = false;
|
isReceivingSynchronous = false;
|
||||||
|
if (messageHandlers.size() > 0) {
|
||||||
|
startReceiveThreadIfRequired();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,12 @@
|
|||||||
|
package org.asamk.signal.manager.api;
|
||||||
|
|
||||||
|
public class AlreadyReceivingException extends Exception {
|
||||||
|
|
||||||
|
public AlreadyReceivingException(String message) {
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
|
||||||
|
public AlreadyReceivingException(String message, Exception e) {
|
||||||
|
super(message, e);
|
||||||
|
}
|
||||||
|
}
|
@ -10,8 +10,10 @@ import org.asamk.signal.OutputType;
|
|||||||
import org.asamk.signal.ReceiveMessageHandler;
|
import org.asamk.signal.ReceiveMessageHandler;
|
||||||
import org.asamk.signal.commands.exceptions.CommandException;
|
import org.asamk.signal.commands.exceptions.CommandException;
|
||||||
import org.asamk.signal.commands.exceptions.IOErrorException;
|
import org.asamk.signal.commands.exceptions.IOErrorException;
|
||||||
|
import org.asamk.signal.commands.exceptions.UserErrorException;
|
||||||
import org.asamk.signal.json.JsonReceiveMessageHandler;
|
import org.asamk.signal.json.JsonReceiveMessageHandler;
|
||||||
import org.asamk.signal.manager.Manager;
|
import org.asamk.signal.manager.Manager;
|
||||||
|
import org.asamk.signal.manager.api.AlreadyReceivingException;
|
||||||
import org.asamk.signal.manager.api.ReceiveConfig;
|
import org.asamk.signal.manager.api.ReceiveConfig;
|
||||||
import org.asamk.signal.output.JsonWriter;
|
import org.asamk.signal.output.JsonWriter;
|
||||||
import org.asamk.signal.output.OutputWriter;
|
import org.asamk.signal.output.OutputWriter;
|
||||||
@ -79,6 +81,8 @@ public class ReceiveCommand implements LocalCommand, JsonRpcSingleCommand<Receiv
|
|||||||
m.receiveMessages(Optional.ofNullable(duration), Optional.ofNullable(maxMessages), handler);
|
m.receiveMessages(Optional.ofNullable(duration), Optional.ofNullable(maxMessages), handler);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new IOErrorException("Error while receiving messages: " + e.getMessage(), e);
|
throw new IOErrorException("Error while receiving messages: " + e.getMessage(), e);
|
||||||
|
} catch (AlreadyReceivingException e) {
|
||||||
|
throw new UserErrorException("Receive command cannot be used if messages are already being received.", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -103,6 +107,8 @@ public class ReceiveCommand implements LocalCommand, JsonRpcSingleCommand<Receiv
|
|||||||
jsonWriter.write(messages);
|
jsonWriter.write(messages);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new IOErrorException("Error while receiving messages: " + e.getMessage(), e);
|
throw new IOErrorException("Error while receiving messages: " + e.getMessage(), e);
|
||||||
|
} catch (AlreadyReceivingException e) {
|
||||||
|
throw new UserErrorException("Receive command cannot be used if messages are already being received.", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user