org.praxislive.code.SharedCodeService Maven / Gradle / Ivy
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright 2024 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.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.stream.Stream;
import org.praxislive.core.ControlAddress;
import org.praxislive.core.ControlInfo;
import org.praxislive.core.services.LogBuilder;
import org.praxislive.core.services.LogLevel;
import org.praxislive.core.services.Service;
import org.praxislive.core.types.PMap;
import org.praxislive.core.types.PReference;
/**
* A {@link Service} for handling shared code updates and creating dependent
* {@link CodeContext}. Must be running in the same process as the components
* due to task and result references. Should make use of a
* {@link CodeCompilerService} implementation for compiling source code (which
* does support other processes).
*/
public final class SharedCodeService implements Service {
/**
* Control ID of the new shared code control.
*/
public final static String NEW_SHARED = "new-shared";
/**
* ControlInfo of the new shared code control.
*/
public final static ControlInfo NEW_SHARED_INFO
= ControlInfo.createFunctionInfo(
List.of(PReference.info(Task.class)),
List.of(PReference.info(Result.class)),
PMap.EMPTY);
@Override
public Stream controls() {
return Stream.of(NEW_SHARED);
}
@Override
public ControlInfo getControlInfo(String control) {
if (NEW_SHARED.equals(control)) {
return NEW_SHARED_INFO;
}
throw new IllegalArgumentException();
}
/**
* Task containing new shared code and dependents to be updated, for sending
* to the SharedCodeService.
*/
public static class Task {
private final PMap sources;
private final Map> dependents;
private final LogLevel logLevel;
/**
* Create a Task.
*
* @param sources new shared code sources
* @param dependents map of dependent tasks by address
* @param logLevel logging level
*/
public Task(PMap sources,
Map> dependents,
LogLevel logLevel) {
this.sources = Objects.requireNonNull(sources);
this.dependents = Map.copyOf(dependents);
this.logLevel = logLevel;
}
/**
* Get the shared code sources.
*
* @return sources
*/
public PMap getSources() {
return sources;
}
/**
* Get the map of dependents.
*
* @return dependents
*/
public Map> getDependents() {
return dependents;
}
/**
* Get the logging level.
*
* @return logging level
*/
public LogLevel getLogLevel() {
return logLevel;
}
}
/**
* Result with shared classes, dependent code contexts, and log.
*/
public static class Result {
private final ClassLoader sharedClasses;
private final Map> dependents;
private final LogBuilder log;
/**
* Create an empty Result.
*/
public Result() {
this.sharedClasses = null;
this.dependents = Map.of();
this.log = new LogBuilder(LogLevel.ERROR);
}
/**
* Create a Result.
*
* @param sharedClasses new shared classes classloader
* @param dependents map of dependent results
* @param log log
*/
public Result(ClassLoader sharedClasses,
Map> dependents,
LogBuilder log) {
this.sharedClasses = Objects.requireNonNull(sharedClasses);
this.dependents = Map.copyOf(dependents);
this.log = Objects.requireNonNull(log);
}
/**
* Get the shared classes classloader.
*
* @return shared classes
*/
public ClassLoader getSharedClasses() {
return sharedClasses;
}
/**
* Get the map if dependent results.
*
* @return dependents
*/
public Map> getDependents() {
return dependents;
}
/**
* Get the log
*
* @return log
*/
public LogBuilder getLog() {
return log;
}
}
/**
* A dependent task for recompiling a {@link CodeDelegate} against the new
* shared code classes.
*
* @param base delegate type
*/
public static class DependentTask {
private final CodeFactory factory;
private final String existingSource;
private final Class existingClass;
/**
* Create a DependentTask.
*
* @param factory code factory for delegate
* @param existingSource existing source to recompile
* @param existingClass existing class
*/
public DependentTask(CodeFactory factory,
String existingSource,
Class existingClass) {
this.factory = Objects.requireNonNull(factory);
this.existingSource = Objects.requireNonNull(existingSource);
this.existingClass = Objects.requireNonNull(existingClass);
}
/**
* Get code factory.
*
* @return code factory
*/
public CodeFactory getFactory() {
return factory;
}
/**
* Get the existing source.
*
* @return existing source
*/
public String getExistingSource() {
return existingSource;
}
/**
* Get the existing class.
*
* @return existing class
*/
public Class getExistingClass() {
return existingClass;
}
}
/**
* A dependent result with new code context linked to new shared code
* classes.
*
* @param base delegate type
*/
public static class DependentResult {
private final CodeContext context;
private final Class existing;
/**
* Create a DependentResult.
*
* @param context code context
* @param existing the existing (previous) class
*/
public DependentResult(CodeContext context,
Class existing) {
this.context = Objects.requireNonNull(context);
this.existing = Objects.requireNonNull(existing);
}
/**
* Get the code context.
*
* @return code context
*/
public CodeContext getContext() {
return context;
}
/**
* Get the existing (previous) class. Used for validation.
*
* @return existing (previous) class
*/
public Class extends D> getExisting() {
return existing;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy