All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.praxislive.code.ResponseHandler Maven / Gradle / Ivy

Go to download

Forest-of-actors runtime supporting real-time systems and real-time recoding - bringing aspects of Erlang, Smalltalk and Extempore to Java.

The newest version!
/*
 * 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;
            }
        }

    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy