de.learnlib.algorithms.dhc.mealy.MealyDHC Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of learnlib-dhc Show documentation
Show all versions of learnlib-dhc Show documentation
The Direct Hypothesis Construction algorithm for active learning of Mealy machines
/* Copyright (C) 2013-2018 TU Dortmund
* This file is part of LearnLib, http://www.learnlib.de/.
*
* 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 de.learnlib.algorithms.dhc.mealy;
import java.io.Serializable;
import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Queue;
import com.github.misberner.buildergen.annotations.GenerateBuilder;
import com.google.common.collect.Interner;
import com.google.common.collect.Interners;
import com.google.common.collect.Sets;
import de.learnlib.api.AccessSequenceTransformer;
import de.learnlib.api.algorithm.LearningAlgorithm.MealyLearner;
import de.learnlib.api.algorithm.feature.GlobalSuffixLearner.GlobalSuffixLearnerMealy;
import de.learnlib.api.algorithm.feature.ResumableLearner;
import de.learnlib.api.algorithm.feature.SupportsGrowingAlphabet;
import de.learnlib.api.oracle.MembershipOracle;
import de.learnlib.api.query.DefaultQuery;
import de.learnlib.counterexamples.GlobalSuffixFinder;
import de.learnlib.counterexamples.GlobalSuffixFinders;
import net.automatalib.automata.transout.impl.compact.CompactMealy;
import net.automatalib.commons.util.mappings.MapMapping;
import net.automatalib.commons.util.mappings.MutableMapping;
import net.automatalib.words.Alphabet;
import net.automatalib.words.Word;
import net.automatalib.words.impl.Alphabets;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* @author Maik Merten
*/
public class MealyDHC implements MealyLearner,
AccessSequenceTransformer,
GlobalSuffixLearnerMealy,
SupportsGrowingAlphabet,
ResumableLearner> {
private static final Logger LOG = LoggerFactory.getLogger(MealyDHC.class);
private final MembershipOracle> oracle;
private Alphabet alphabet;
private LinkedHashSet> splitters = new LinkedHashSet<>();
private CompactMealy hypothesis;
private MutableMapping> accessSequences;
private final GlobalSuffixFinder super I, ? super Word> suffixFinder;
/**
* Constructor, provided for backwards compatibility reasons.
*
* @param alphabet
* the learning alphabet
* @param oracle
* the learning membership oracle
*/
public MealyDHC(Alphabet alphabet, MembershipOracle> oracle) {
this(alphabet, oracle, GlobalSuffixFinders.RIVEST_SCHAPIRE, null);
}
/**
* Constructor.
*
* @param alphabet
* the learning alphabet
* @param oracle
* the learning membership oracle
* @param suffixFinder
* the {@link GlobalSuffixFinder suffix finder} to use for analyzing counterexamples
* @param initialSplitters
* the initial set of splitters, {@code null} or an empty collection will result in the set of splitters
* being initialized as the set of alphabet symbols (interpreted as {@link Word}s)
*/
@GenerateBuilder(defaults = BuilderDefaults.class, builderFinal = false)
public MealyDHC(Alphabet alphabet,
MembershipOracle> oracle,
GlobalSuffixFinder super I, ? super Word> suffixFinder,
Collection extends Word> initialSplitters) {
this.alphabet = alphabet;
this.oracle = oracle;
this.suffixFinder = suffixFinder;
// ensure that the first k splitters are the k alphabet symbols,
// in correct order (this is required by scheduleSuccessors)
for (I symbol : alphabet) {
splitters.add(Word.fromLetter(symbol));
}
if (initialSplitters != null) {
splitters.addAll(initialSplitters);
}
}
@Override
public Collection> getGlobalSuffixes() {
return Collections.unmodifiableCollection(splitters);
}
@Override
public boolean addGlobalSuffixes(Collection extends Word> newGlobalSuffixes) {
checkInternalState();
return addSuffixesUnchecked(newGlobalSuffixes);
}
private void checkInternalState() {
if (hypothesis == null) {
throw new IllegalStateException("No hypothesis learned yet");
}
}
protected boolean addSuffixesUnchecked(Collection extends Word> newSuffixes) {
int oldSize = hypothesis.size();
for (Word suf : newSuffixes) {
if (!splitters.contains(suf)) {
splitters.add(suf);
LOG.debug("added suffix: {0}", suf);
}
}
startLearning();
return (hypothesis.size() != oldSize);
}
@Override
public void startLearning() {
// initialize structure to store state output signatures
Map>, Integer> signatures = new HashMap<>();
// set up new hypothesis machine
hypothesis = new CompactMealy<>(alphabet);
// initialize exploration queue
Queue> queue = new ArrayDeque<>();
// initialize storage for access sequences
accessSequences = hypothesis.createDynamicStateMapping();
// first element to be explored represents the initial state with no predecessor
queue.add(new QueueElement<>(null, null, null, null));
Interner> deduplicator = Interners.newStrongInterner();
while (!queue.isEmpty()) {
// get element to be explored from queue
QueueElement elem = queue.poll();
// determine access sequence for state
Word access = assembleAccessSequence(elem);
// assemble queries
ArrayList>> queries = new ArrayList<>(splitters.size());
for (Word suffix : splitters) {
queries.add(new DefaultQuery<>(access, suffix));
}
// retrieve answers
oracle.processQueries(queries);
// assemble output signature
List> sig = new ArrayList<>(splitters.size());
for (DefaultQuery> query : queries) {
sig.add(deduplicator.intern(query.getOutput()));
}
Integer sibling = signatures.get(sig);
if (sibling != null) {
// this element does not possess a new output signature
// create a transition from parent state to sibling
hypothesis.addTransition(elem.parentState, elem.transIn, sibling, elem.transOut);
} else {
// this is actually an observably distinct state! Progress!
// Create state and connect via transition to parent
Integer state = elem.parentElement == null ? hypothesis.addInitialState() : hypothesis.addState();
if (elem.parentElement != null) {
hypothesis.addTransition(elem.parentState, elem.transIn, state, elem.transOut);
}
signatures.put(sig, state);
accessSequences.put(state, elem);
scheduleSuccessors(elem, state, queue, sig);
}
}
}
private Word assembleAccessSequence(QueueElement elem) {
List word = new ArrayList<>(elem.depth);
QueueElement pre = elem.parentElement;
I sym = elem.transIn;
while (pre != null && sym != null) {
word.add(sym);
sym = pre.transIn;
pre = pre.parentElement;
}
Collections.reverse(word);
return Word.fromList(word);
}
private void scheduleSuccessors(QueueElement elem,
Integer state,
Queue> queue,
List> sig) throws IllegalArgumentException {
for (int i = 0; i < alphabet.size(); ++i) {
// retrieve I/O for transition
I input = alphabet.getSymbol(i);
O output = sig.get(i).getSymbol(0);
// create successor element and schedule for exploration
queue.add(new QueueElement<>(state, elem, input, output));
}
}
@Override
public boolean refineHypothesis(DefaultQuery> ceQuery) {
checkInternalState();
Collection> ceSuffixes = suffixFinder.findSuffixes(ceQuery, this, hypothesis, oracle);
return addSuffixesUnchecked(ceSuffixes);
}
@Override
public CompactMealy getHypothesisModel() {
checkInternalState();
return hypothesis;
}
@Override
public void addAlphabetSymbol(I symbol) {
if (this.alphabet.containsSymbol(symbol)) {
return;
}
final Iterator> splitterIterator = this.splitters.iterator();
final LinkedHashSet> newSplitters = Sets.newLinkedHashSetWithExpectedSize(this.splitters.size() + 1);
// see initial initialization of the splitters
for (int i = 0; i < this.alphabet.size(); i++) {
newSplitters.add(splitterIterator.next());
}
newSplitters.add(Word.fromLetter(symbol));
while (splitterIterator.hasNext()) {
newSplitters.add(splitterIterator.next());
}
this.alphabet = Alphabets.withNewSymbol(this.alphabet, symbol);
this.splitters = newSplitters;
this.startLearning();
}
@Override
public MealyDHCState suspend() {
return new MealyDHCState<>(splitters, hypothesis, accessSequences);
}
@Override
public void resume(final MealyDHCState state) {
this.splitters = state.getSplitters();
this.accessSequences = new MapMapping<>(state.getAccessSequences());
this.hypothesis = state.getHypothesis();
}
@Override
public Word transformAccessSequence(Word word) {
checkInternalState();
Integer state = hypothesis.getSuccessor(hypothesis.getInitialState(), word);
return assembleAccessSequence(accessSequences.get(state));
}
@Override
public boolean isAccessSequence(Word word) {
checkInternalState();
Word canonical = transformAccessSequence(word);
return canonical.equals(word);
}
public static class BuilderDefaults {
public static GlobalSuffixFinder super I, ? super Word> suffixFinder() {
return GlobalSuffixFinders.RIVEST_SCHAPIRE;
}
public static Collection> initialSplitters() {
return null;
}
}
static final class QueueElement implements Serializable {
private final Integer parentState;
private final QueueElement parentElement;
private final I transIn;
private final O transOut;
private final int depth;
private QueueElement(Integer parentState, QueueElement parentElement, I transIn, O transOut) {
this.parentState = parentState;
this.parentElement = parentElement;
this.transIn = transIn;
this.transOut = transOut;
this.depth = (parentElement != null) ? parentElement.depth + 1 : 0;
}
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy