de.learnlib.oracle.equivalence.RandomWpMethodEQOracle Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of learnlib-equivalence-oracles Show documentation
Show all versions of learnlib-equivalence-oracles Show documentation
A collection of equivalence oracles
/* 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.oracle.equivalence;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Random;
import java.util.stream.Stream;
import de.learnlib.api.oracle.MembershipOracle;
import net.automatalib.automata.UniversalDeterministicAutomaton;
import net.automatalib.automata.concepts.Output;
import net.automatalib.automata.fsa.DFA;
import net.automatalib.automata.transout.MealyMachine;
import net.automatalib.commons.util.mappings.MutableMapping;
import net.automatalib.util.automata.Automata;
import net.automatalib.util.automata.cover.Covers;
import net.automatalib.words.Word;
import net.automatalib.words.WordBuilder;
/**
* Implements an equivalence test by applying the Wp-method test on the given hypothesis automaton, as described in
* "Test Selection Based on Finite State Models" by S. Fujiwara et al. Instead of enumerating the test suite in order,
* this is a sampling implementation: 1. sample uniformly from the states for a prefix 2. sample geometrically a random
* word 3. sample a word from the set of suffixes / state identifiers (either local or global) There are two parameters:
* minimalSize determines the minimal size of the random word, this is useful when one first performs a W(p)-method with
* some depth and continue with this randomized tester from that depth onward. The second parameter rndLength determines
* the expected length of the random word. (The expected length in effect is minimalSize + rndLength.) In the unbounded
* case it will not terminate for a correct hypothesis.
*
* @param
* automaton type
* @param
* input symbol type
* @param
* output domain type
*
* @author Joshua Moerman
*/
public class RandomWpMethodEQOracle & Output, I, D>
extends AbstractTestWordEQOracle {
private final int minimalSize;
private final int rndLength;
private final int bound;
private final Random rand;
/**
* Constructor for an unbounded testing oracle.
*
* @param sulOracle
* oracle which answers tests.
* @param minimalSize
* minimal size of the random word
* @param rndLength
* expected length (in addition to minimalSize) of random word
*/
public RandomWpMethodEQOracle(MembershipOracle sulOracle, int minimalSize, int rndLength) {
this(sulOracle, minimalSize, rndLength, 0);
}
/**
* Constructor for a bounded testing oracle.
*
* @param sulOracle
* oracle which answers tests.
* @param minimalSize
* minimal size of the random word
* @param rndLength
* expected length (in addition to minimalSize) of random word
* @param bound
* specifies the bound (set to 0 for unbounded).
*/
public RandomWpMethodEQOracle(MembershipOracle sulOracle, int minimalSize, int rndLength, int bound) {
this(sulOracle, minimalSize, rndLength, bound, 1);
}
/**
* Constructor for a bounded testing oracle with specific batch size.
*
* @param sulOracle
* oracle which answers tests.
* @param minimalSize
* minimal size of the random word
* @param rndLength
* expected length (in addition to minimalSize) of random word
* @param bound
* specifies the bound (set to 0 for unbounded).
* @param batchSize
* size of the batches sent to the membership oracle
*/
public RandomWpMethodEQOracle(MembershipOracle sulOracle,
int minimalSize,
int rndLength,
int bound,
int batchSize) {
super(sulOracle, batchSize);
this.minimalSize = minimalSize;
this.rndLength = rndLength;
this.bound = bound;
this.rand = new Random();
}
/**
* Constructor for a bounded testing oracle with specific batch size.
*
* @param sulOracle
* oracle which answers tests.
* @param minimalSize
* minimal size of the random word
* @param rndLength
* expected length (in addition to minimalSize) of random word
* @param bound
* specifies the bound (set to 0 for unbounded).
* @param random
* custom Random generator.
* @param batchSize
* size of the batches sent to the membership oracle
*/
public RandomWpMethodEQOracle(MembershipOracle sulOracle,
int minimalSize,
int rndLength,
int bound,
Random random,
int batchSize) {
super(sulOracle, batchSize);
this.minimalSize = minimalSize;
this.rndLength = rndLength;
this.bound = bound;
this.rand = random;
}
@Override
protected Stream> generateTestWords(A hypothesis, Collection extends I> inputs) {
UniversalDeterministicAutomaton, I, ?, ?, ?> aut = hypothesis;
return doGenerateTestWords(aut, inputs);
}
protected Stream> doGenerateTestWords(UniversalDeterministicAutomaton hypothesis,
Collection extends I> inputs) {
// Note that we want to use ArrayLists because we want constant time random access
// We will sample from this for a prefix
ArrayList> stateCover = new ArrayList<>(hypothesis.size());
Covers.stateCover(hypothesis, inputs, stateCover);
// Then repeatedly from this for a random word
ArrayList arrayAlphabet = new ArrayList<>(inputs);
// Finally we test the state with a suffix, sometimes a global one, sometimes local
ArrayList> globalSuffixes = new ArrayList<>();
Automata.characterizingSet(hypothesis, inputs, globalSuffixes);
MutableMapping>> localSuffixSets = hypothesis.createStaticStateMapping();
for (S state : hypothesis.getStates()) {
ArrayList> suffixSet = new ArrayList<>();
Automata.stateCharacterizingSet(hypothesis, inputs, state, suffixSet);
localSuffixSets.put(state, suffixSet);
}
final Stream> result = Stream.generate(() -> generateSingleTestWord(hypothesis,
stateCover,
arrayAlphabet,
globalSuffixes,
localSuffixSets));
return bound > 0 ? result.limit(bound) : result;
}
private Word generateSingleTestWord(UniversalDeterministicAutomaton hypothesis,
ArrayList> stateCover,
ArrayList arrayAlphabet,
ArrayList> globalSuffixes,
MutableMapping>> localSuffixSets) {
WordBuilder wb = new WordBuilder<>(minimalSize + rndLength + 1);
// pick a random state
wb.append(stateCover.get(rand.nextInt(stateCover.size())));
// construct random middle part (of some expected length)
int size = minimalSize;
while ((size > 0) || (rand.nextDouble() > 1 / (rndLength + 1.0))) {
wb.append(arrayAlphabet.get(rand.nextInt(arrayAlphabet.size())));
if (size > 0) {
size--;
}
}
// pick a random suffix for this state
// 50% chance for state testing, 50% chance for transition testing
if (rand.nextBoolean()) {
// global
if (!globalSuffixes.isEmpty()) {
wb.append(globalSuffixes.get(rand.nextInt(globalSuffixes.size())));
}
} else {
// local
S state2 = hypothesis.getState(wb);
ArrayList> localSuffixes = localSuffixSets.get(state2);
if (!localSuffixes.isEmpty()) {
wb.append(localSuffixes.get(rand.nextInt(localSuffixes.size())));
}
}
return wb.toWord();
}
public static class DFARandomWpMethodEQOracle
extends RandomWpMethodEQOracle, I, Boolean>
implements DFAEquivalenceOracle {
public DFARandomWpMethodEQOracle(MembershipOracle mqOracle,
int minimalSize,
int rndLength) {
super(mqOracle, minimalSize, rndLength);
}
public DFARandomWpMethodEQOracle(MembershipOracle mqOracle,
int minimalSize,
int rndLength,
int bound) {
super(mqOracle, minimalSize, rndLength, bound);
}
public DFARandomWpMethodEQOracle(MembershipOracle mqOracle,
int minimalSize,
int rndLength,
int bound,
int batchSize) {
super(mqOracle, minimalSize, rndLength, bound, batchSize);
}
public DFARandomWpMethodEQOracle(MembershipOracle mqOracle,
int minimalSize,
int rndLength,
int bound,
Random random,
int batchSize) {
super(mqOracle, minimalSize, rndLength, bound, random, batchSize);
}
}
public static class MealyRandomWpMethodEQOracle
extends RandomWpMethodEQOracle, I, Word>
implements MealyEquivalenceOracle {
public MealyRandomWpMethodEQOracle(MembershipOracle> mqOracle,
int minimalSize,
int rndLength) {
super(mqOracle, minimalSize, rndLength);
}
public MealyRandomWpMethodEQOracle(MembershipOracle> mqOracle,
int minimalSize,
int rndLength,
int bound) {
super(mqOracle, minimalSize, rndLength, bound);
}
public MealyRandomWpMethodEQOracle(MembershipOracle> mqOracle,
int minimalSize,
int rndLength,
int bound,
int batchSize) {
super(mqOracle, minimalSize, rndLength, bound, batchSize);
}
public MealyRandomWpMethodEQOracle(MembershipOracle> mqOracle,
int minimalSize,
int rndLength,
int bound,
Random random,
int batchSize) {
super(mqOracle, minimalSize, rndLength, bound, random, batchSize);
}
}
}