/*
* Copyright 2003-2016 JetBrains s.r.o.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package jetbrains.mps.textgen.trace;
import jetbrains.mps.smodel.ModelAccessHelper;
import jetbrains.mps.smodel.SModelStereotype;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.mps.openapi.model.SModel;
import org.jetbrains.mps.openapi.model.SModelName;
import org.jetbrains.mps.openapi.module.SRepository;
import java.util.Arrays;
import java.util.function.Predicate;
import java.util.stream.Stream;
import java.util.stream.StreamSupport;
/**
* Generic implementation that looks for named models inside a repository and their associated {@link DebugInfo}.
* This provider ignores stub model (doesn't expect stub model to be a source of generation).
*
* @author Artem Tikhomirov
* @since 3.4
*/
public class DefaultTraceInfoProvider implements TraceInfoProvider {
private final SRepository myRepository;
public DefaultTraceInfoProvider(@NotNull SRepository repository) {
myRepository = repository;
}
@Override
public Stream<DebugInfo> debugInfo(@NotNull SModelName modelName, @NotNull Predicate<SModel> modelFilter) {
// if exact model requested, find the one. if no stereotype specified, find any matching by long name only
Predicate<SModel> nameCheck = modelName.hasStereotype() ? m -> modelName.equals(m.getName()) : m -> modelName.getLongName().equals(m.getName().getLongName());
// XXX can cache DebugInfo[] for given model name if lookup turns out slow, just don't want optimize early.
// If cache, shall specify whether this provider listens to changes in TraceInfoCache or not. generally, shall not as caller has a chance to control
// lifecycle of the provider and I expect uses of this class to be either short-lived or bound to another object (e.g. debug session) where it's
// reasonable to expect no changes like removal of TraceInfoCache.
Predicate<SModel> notStubModelWithTrace = m -> !SModelStereotype.isStubModel(m) && TraceInfo.hasTrace(m);
// XXX need a fast way to iterate over names of models accessible in a repository
final DebugInfo[] debugInfos = new ModelAccessHelper(myRepository).runReadAction(() ->
StreamSupport.stream(myRepository.getModules().spliterator(), false).flatMap(
module -> StreamSupport.stream(module.getModels().spliterator(), false)).filter(
nameCheck.and(notStubModelWithTrace).and(modelFilter)).map(m -> TraceInfoCache.getInstance().get(m)).toArray(DebugInfo[]::new)
);
return Arrays.stream(debugInfos);
}
}