org.praxislive.code.ResponseHandler Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of praxiscore-code Show documentation
Show all versions of praxiscore-code Show documentation
Forest-of-actors runtime supporting real-time systems and real-time recoding - bringing aspects of Erlang, Smalltalk and Extempore to Java.
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright 2023 Neil C Smith.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License version 3 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
* version 3 for more details.
*
* You should have received a copy of the GNU Lesser General Public License version 3
* along with this work; if not, see http://www.gnu.org/licenses/
*
*
* Please visit https://www.praxislive.org if you need additional information or
* have any questions.
*/
package org.praxislive.code;
import java.lang.ref.WeakReference;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import org.praxislive.code.userapi.Async;
import org.praxislive.core.Call;
import org.praxislive.core.Control;
import org.praxislive.core.ControlInfo;
import org.praxislive.core.PacketRouter;
import org.praxislive.core.Value;
import org.praxislive.core.services.LogLevel;
import org.praxislive.core.types.PError;
/**
*
*/
class ResponseHandler extends ControlDescriptor implements Control {
static final String ID = "_reply";
private static final PError UNKNOWN_ERROR = PError.of("Unknown error");
private final Map> resultMap;
private CodeContext> context;
ResponseHandler(int index) {
super(ResponseHandler.class, ID, Category.Internal, index);
this.resultMap = new HashMap<>();
}
@Override
public void attach(CodeContext> context, ResponseHandler previous) {
if (previous != null) {
resultMap.putAll(previous.resultMap);
previous.resultMap.clear();
}
this.context = context;
}
@Override
public void call(Call call, PacketRouter router) throws Exception {
if (call.isReply()) {
var asyncRef = resultMap.remove(call.matchID());
if (asyncRef != null) {
asyncRef.complete(call);
}
} else if (call.isError()) {
var error = extractError(call.args());
var asyncRef = resultMap.remove(call.matchID());
if (asyncRef == null || !asyncRef.completeWithError(error)) {
context.getLog().log(LogLevel.ERROR, error);
}
} else if (call.isReplyRequired()) {
router.route(call.error(PError.of("Unexpected call")));
}
cleanResultMap();
}
@Override
public Control control() {
return this;
}
@Override
public ControlInfo controlInfo() {
return null;
}
@Override
public void dispose() {
resultMap.forEach((id, ref) -> {
Async> async = ref.get();
if (async != null) {
async.fail(PError.of("Disposed"));
}
});
resultMap.clear();
}
void register(Call call, Async async) {
register(call, async, Function.identity());
}
void register(Call call, Async async, Function converter) {
cleanResultMap();
resultMap.put(call.matchID(), new AsyncReference(async, converter));
}
private void cleanResultMap() {
resultMap.entrySet().removeIf(e -> e.getValue().get() == null);
}
private PError extractError(List args) {
if (args.isEmpty()) {
return UNKNOWN_ERROR;
} else {
return PError.from(args.get(0))
.orElse(PError.of(args.get(0).toString()));
}
}
private static class AsyncReference extends WeakReference> {
private final Function converter;
private AsyncReference(Async referent, Function converter) {
super(referent);
this.converter = converter;
}
private void complete(Call call) {
Async async = get();
if (async != null) {
try {
T value = converter.apply(call);
async.complete(value);
} catch (Exception ex) {
async.fail(PError.of(ex));
}
}
}
private boolean completeWithError(PError error) {
Async async = get();
if (async != null) {
return async.fail(error);
} else {
return false;
}
}
}
}