diff --git a/inception/inception-concept-linking/src/main/java/de/tudarmstadt/ukp/inception/conceptlinking/recommender/NamedEntityLinker.java b/inception/inception-concept-linking/src/main/java/de/tudarmstadt/ukp/inception/conceptlinking/recommender/NamedEntityLinker.java index 1ec839a07da..73d26263448 100644 --- a/inception/inception-concept-linking/src/main/java/de/tudarmstadt/ukp/inception/conceptlinking/recommender/NamedEntityLinker.java +++ b/inception/inception-concept-linking/src/main/java/de/tudarmstadt/ukp/inception/conceptlinking/recommender/NamedEntityLinker.java @@ -43,6 +43,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.DataSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -90,7 +91,7 @@ public void train(RecommenderContext aContext, List aCasList) } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { Type predictedType = getPredictedType(aCas); diff --git a/inception/inception-concept-linking/src/test/java/de/tudarmstadt/ukp/inception/conceptlinking/NamedEntityLinkerTest.java b/inception/inception-concept-linking/src/test/java/de/tudarmstadt/ukp/inception/conceptlinking/NamedEntityLinkerTest.java index 8f5f49ba4ab..0fbe22f3649 100644 --- a/inception/inception-concept-linking/src/test/java/de/tudarmstadt/ukp/inception/conceptlinking/NamedEntityLinkerTest.java +++ b/inception/inception-concept-linking/src/test/java/de/tudarmstadt/ukp/inception/conceptlinking/NamedEntityLinkerTest.java @@ -65,6 +65,7 @@ import de.tudarmstadt.ukp.inception.kb.graph.KBHandle; import de.tudarmstadt.ukp.inception.kb.model.KnowledgeBase; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.schema.api.feature.FeatureSupport; import de.tudarmstadt.ukp.inception.schema.api.feature.FeatureSupportRegistry; @@ -146,7 +147,7 @@ public void thatPredictionWorks() throws Exception sut.train(context, Collections.singletonList(cas)); RecommenderTestHelper.addScoreFeature(cas, NamedEntity.class, "value"); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); List predictions = getPredictions(cas, NamedEntity.class); diff --git a/inception/inception-diam-compactv2/src/main/java/de/tudarmstadt/ukp/inception/diam/model/compactv2/CompactAnnotationAttributes.java b/inception/inception-diam-compactv2/src/main/java/de/tudarmstadt/ukp/inception/diam/model/compactv2/CompactAnnotationAttributes.java index 22868dd035d..bb88495acbe 100644 --- a/inception/inception-diam-compactv2/src/main/java/de/tudarmstadt/ukp/inception/diam/model/compactv2/CompactAnnotationAttributes.java +++ b/inception/inception-diam-compactv2/src/main/java/de/tudarmstadt/ukp/inception/diam/model/compactv2/CompactAnnotationAttributes.java @@ -20,6 +20,8 @@ import java.util.ArrayList; import java.util.List; +import com.fasterxml.jackson.annotation.JsonFormat; +import com.fasterxml.jackson.annotation.JsonFormat.Shape; import com.fasterxml.jackson.annotation.JsonInclude; import com.fasterxml.jackson.annotation.JsonInclude.Include; import com.fasterxml.jackson.annotation.JsonProperty; @@ -32,11 +34,13 @@ public class CompactAnnotationAttributes public static final String ATTR_COLOR = "c"; public static final String ATTR_COMMENTS = "cm"; public static final String ATTR_SCORE = "s"; + public static final String ATTR_HIDE_SCORE = "hs"; private @JsonProperty(ATTR_LABEL) String labelText; private @JsonProperty(ATTR_COLOR) String color; private @JsonProperty(ATTR_COMMENTS) List comments; private @JsonProperty(ATTR_SCORE) double score; + private @JsonProperty(ATTR_HIDE_SCORE) boolean hideScore; @JsonInclude(Include.NON_DEFAULT) @JsonSerialize(using = ScoreSerializer.class) @@ -50,6 +54,18 @@ public void setScore(double aScore) score = aScore; } + @JsonFormat(shape = Shape.NUMBER) + @JsonInclude(Include.NON_DEFAULT) + public boolean isHideScore() + { + return hideScore; + } + + public void setHideScore(boolean aHideScore) + { + hideScore = aHideScore; + } + public void setLabelText(String aLabelText) { labelText = aLabelText; diff --git a/inception/inception-diam-compactv2/src/main/java/de/tudarmstadt/ukp/inception/diam/model/compactv2/CompactSerializerV2Impl.java b/inception/inception-diam-compactv2/src/main/java/de/tudarmstadt/ukp/inception/diam/model/compactv2/CompactSerializerV2Impl.java index e327adec082..b90b0cd26bf 100644 --- a/inception/inception-diam-compactv2/src/main/java/de/tudarmstadt/ukp/inception/diam/model/compactv2/CompactSerializerV2Impl.java +++ b/inception/inception-diam-compactv2/src/main/java/de/tudarmstadt/ukp/inception/diam/model/compactv2/CompactSerializerV2Impl.java @@ -126,6 +126,7 @@ private CompactRelation renderRelation(VArc varc) getArgument(varc.getSource(), varc.getTarget()), varc.getLabelHint(), varc.getColorHint()); carc.getAttributes().setScore(varc.getScore()); + carc.getAttributes().setHideScore(varc.isHideScore()); return carc; } @@ -152,6 +153,7 @@ private CompactSpan renderSpan(RenderRequest aRequest, VSpan vspan) vspan.getColorHint()); } cspan.getAttributes().setScore(vspan.getScore()); + cspan.getAttributes().setHideScore(vspan.isHideScore()); return cspan; } diff --git a/inception/inception-diam-editor/src/main/ts/src/LabelBadge.svelte b/inception/inception-diam-editor/src/main/ts/src/LabelBadge.svelte index 0e28e795d8e..12d8158f929 100644 --- a/inception/inception-diam-editor/src/main/ts/src/LabelBadge.svelte +++ b/inception/inception-diam-editor/src/main/ts/src/LabelBadge.svelte @@ -61,7 +61,8 @@ {#if showText} {renderLabel(annotation)} {/if} - {#if annotation.score} + + {#if annotation.score && !annotation.hideScore} {annotation.score.toFixed(2)} @@ -88,7 +89,8 @@ {#if showText} {renderLabel(annotation)} {/if} - {#if annotation.score} + + {#if annotation.score && !annotation.hideScore} {annotation.score.toFixed(2)} diff --git a/inception/inception-example-imls-data-majority/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/datamajority/DataMajorityNerRecommender.java b/inception/inception-example-imls-data-majority/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/datamajority/DataMajorityNerRecommender.java index 879ea387eb5..028d8790828 100644 --- a/inception/inception-example-imls-data-majority/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/datamajority/DataMajorityNerRecommender.java +++ b/inception/inception-example-imls-data-majority/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/datamajority/DataMajorityNerRecommender.java @@ -42,6 +42,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.LabelPair; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -133,7 +134,7 @@ private DataMajorityModel trainModel(List aAnnotations) // tag::predict1[] @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { DataMajorityModel model = aContext.get(KEY_MODEL).orElseThrow( diff --git a/inception/inception-example-imls-data-majority/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/datamajority/DataMajorityNerRecommenderTest.java b/inception/inception-example-imls-data-majority/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/datamajority/DataMajorityNerRecommenderTest.java index 6a8903af83c..ba9e415527a 100644 --- a/inception/inception-example-imls-data-majority/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/datamajority/DataMajorityNerRecommenderTest.java +++ b/inception/inception-example-imls-data-majority/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/datamajority/DataMajorityNerRecommenderTest.java @@ -59,6 +59,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.IncrementalSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.PercentageBasedSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.support.test.recommendation.DkproTestHelper; import de.tudarmstadt.ukp.inception.support.test.recommendation.RecommenderTestHelper; @@ -104,7 +105,7 @@ public void thatPredictionWorks() throws Exception sut.train(context, asList(cas)); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); Collection predictions = getPredictions(cas, NamedEntity.class); diff --git a/inception/inception-imls-elg/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/elg/ElgRecommender.java b/inception/inception-imls-elg/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/elg/ElgRecommender.java index fb16da8c37f..2c5eaa06dff 100644 --- a/inception/inception-imls-elg/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/elg/ElgRecommender.java +++ b/inception/inception-imls-elg/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/elg/ElgRecommender.java @@ -34,8 +34,8 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.NonTrainableRecommenderEngineImplBase; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; -import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.recommendation.imls.elg.model.ElgAnnotation; import de.tudarmstadt.ukp.inception.recommendation.imls.elg.model.ElgAnnotationsResponse; import de.tudarmstadt.ukp.inception.recommendation.imls.elg.model.ElgServiceResponse; @@ -67,7 +67,7 @@ public ElgRecommender(Recommender aRecommender, ElgRecommenderTraits aTraits, } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { ElgServiceResponse response; diff --git a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommender.java b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommender.java index 25bae96ed48..5d389e11cdb 100644 --- a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommender.java +++ b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommender.java @@ -64,6 +64,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.DataSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -216,7 +217,7 @@ else if (response.statusCode() >= HTTP_BAD_REQUEST) { } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { var client = getClient(); diff --git a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderFactory.java b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderFactory.java index 89b392f38e2..c12a7429638 100644 --- a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderFactory.java +++ b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderFactory.java @@ -97,4 +97,11 @@ public boolean isEvaluable() { return false; } + + @Override + public boolean isRanker(Recommender aRecommender) + { + var traits = readTraits(aRecommender); + return traits.isRanker(); + } } diff --git a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraits.java b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraits.java index 80001a4bd25..44925b48422 100644 --- a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraits.java +++ b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraits.java @@ -30,6 +30,7 @@ public class ExternalRecommenderTraits private String remoteUrl; private boolean trainable; private boolean verifyCertificates = true; + private boolean ranker; public String getRemoteUrl() { @@ -60,4 +61,14 @@ public boolean isVerifyCertificates() { return verifyCertificates; } + + public void setRanker(boolean aRanker) + { + ranker = aRanker; + } + + public boolean isRanker() + { + return ranker; + } } diff --git a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.html b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.html index 59a1a5ad665..aa10c097d97 100644 --- a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.html +++ b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.html @@ -47,6 +47,16 @@ +
+
+
+ + +
+
+
diff --git a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.java b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.java index e23e338253e..451c4b4ef53 100644 --- a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.java +++ b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.java @@ -80,6 +80,10 @@ protected void onSubmit() getTrainingStatesChoice().add(visibleWhen(() -> trainable.getModelObject() == true)); + var ranker = new CheckBox("ranker"); + ranker.setOutputMarkupId(true); + form.add(ranker); + add(form); } } diff --git a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.properties b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.properties index 4bd43e422ab..c9fedd4042a 100644 --- a/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.properties +++ b/inception/inception-imls-external/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderTraitsEditor.properties @@ -16,4 +16,5 @@ remoteUrl=Remote URL trainable=Trainable +ranker=Ranker verifyCertificates=Verify certificates diff --git a/inception/inception-imls-external/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderIntegrationTest.java b/inception/inception-imls-external/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderIntegrationTest.java index 742f89695dd..606049ab412 100644 --- a/inception/inception-imls-external/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderIntegrationTest.java +++ b/inception/inception-imls-external/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/ExternalRecommenderIntegrationTest.java @@ -57,6 +57,7 @@ import de.tudarmstadt.ukp.inception.annotation.storage.CasMetadataUtils; import de.tudarmstadt.ukp.inception.annotation.storage.CasStorageSession; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.recommendation.imls.external.v1.config.ExternalRecommenderPropertiesImpl; import de.tudarmstadt.ukp.inception.recommendation.imls.external.v1.messages.PredictionRequest; @@ -133,7 +134,7 @@ public void thatPredictingWorks() throws Exception var cas = casses.get(0); RecommenderTestHelper.addScoreFeature(cas, NamedEntity.class, "value"); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); var predictions = getPredictions(cas, NamedEntity.class); @@ -180,7 +181,7 @@ public void thatPredictingSendsCorrectRequest() throws Exception var cas = casses.get(0); RecommenderTestHelper.addScoreFeature(cas, NamedEntity.class, "value"); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); var request = fromJsonString(PredictionRequest.class, requestBodies.get(1)); diff --git a/inception/inception-imls-external/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/MockRemoteStringMatchingNerRecommender.java b/inception/inception-imls-external/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/MockRemoteStringMatchingNerRecommender.java index 07046edfb16..4a78caf2390 100644 --- a/inception/inception-imls-external/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/MockRemoteStringMatchingNerRecommender.java +++ b/inception/inception-imls-external/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/external/v1/MockRemoteStringMatchingNerRecommender.java @@ -40,6 +40,7 @@ import org.xml.sax.SAXException; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.recommendation.imls.external.v1.messages.PredictionRequest; @@ -93,7 +94,7 @@ public String predict(String aPredictionRequestJson) } } - recommendationEngine.predict(context, cas); + recommendationEngine.predict(new PredictionContext(context), cas); return buildPredictionResponse(cas); } diff --git a/inception/inception-imls-hf/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/hf/HfRecommender.java b/inception/inception-imls-hf/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/hf/HfRecommender.java index 8e2b8c9cf22..5808b0998dc 100644 --- a/inception/inception-imls-hf/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/hf/HfRecommender.java +++ b/inception/inception-imls-hf/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/hf/HfRecommender.java @@ -29,8 +29,8 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.NonTrainableRecommenderEngineImplBase; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; -import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.recommendation.imls.hf.client.HfInferenceClient; import de.tudarmstadt.ukp.inception.recommendation.imls.hf.model.HfEntityGroup; import de.tudarmstadt.ukp.inception.rendering.model.Range; @@ -53,7 +53,7 @@ public HfRecommender(Recommender aRecommender, HfRecommenderTraits aTraits, } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { List response; diff --git a/inception/inception-imls-lapps/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/lapps/LappsGridRecommender.java b/inception/inception-imls-lapps/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/lapps/LappsGridRecommender.java index ac9e0984850..36f69a6a8a2 100644 --- a/inception/inception-imls-lapps/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/lapps/LappsGridRecommender.java +++ b/inception/inception-imls-lapps/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/lapps/LappsGridRecommender.java @@ -39,6 +39,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.DataSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -67,7 +68,7 @@ public void train(RecommenderContext aContext, List aCasses) } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { // FIXME: Ignores begin/end - always fetches predictions for the entire CAS diff --git a/inception/inception-imls-lapps/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/lapps/LappsRecommenderIntegrationTest.java b/inception/inception-imls-lapps/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/lapps/LappsRecommenderIntegrationTest.java index 8259fc3f604..651d3e6048c 100644 --- a/inception/inception-imls-lapps/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/lapps/LappsRecommenderIntegrationTest.java +++ b/inception/inception-imls-lapps/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/lapps/LappsRecommenderIntegrationTest.java @@ -42,6 +42,7 @@ import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; import de.tudarmstadt.ukp.dkpro.core.api.lexmorph.type.pos.POS; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.recommendation.imls.lapps.traits.LappsGridRecommenderTraits; import okhttp3.mockwebserver.MockResponse; @@ -83,7 +84,7 @@ public void thatPredictingPosWorks() throws Exception RecommenderContext context = new RecommenderContext(); CAS cas = loadData(); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); Collection predictions = JCasUtil.select(cas.getJCas(), POS.class); diff --git a/inception/inception-imls-ollama/pom.xml b/inception/inception-imls-ollama/pom.xml index 6df54c51e5b..8feea34cbc1 100644 --- a/inception/inception-imls-ollama/pom.xml +++ b/inception/inception-imls-ollama/pom.xml @@ -46,6 +46,14 @@ de.tudarmstadt.ukp.inception.app inception-api-render + + de.tudarmstadt.ukp.inception.app + inception-api-annotation + + + de.tudarmstadt.ukp.inception.app + inception-layer-docmetadata + de.tudarmstadt.ukp.inception.app inception-support @@ -96,6 +104,10 @@ org.apache.wicket wicket-spring + + com.googlecode.wicket-jquery-ui + wicket-kendo-ui + org.danekja jdk-serializable-functional diff --git a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/ExtractionMode.java b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/ExtractionMode.java index 272b4a2787d..875b6bb0c78 100644 --- a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/ExtractionMode.java +++ b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/ExtractionMode.java @@ -19,11 +19,24 @@ import com.fasterxml.jackson.annotation.JsonProperty; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; +import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanLayerSupport; + public enum ExtractionMode { @JsonProperty("response-as-label") RESPONSE_AS_LABEL, // @JsonProperty("mentions-from-json") - MENTIONS_FROM_JSON + MENTIONS_FROM_JSON; + + public boolean accepts(AnnotationLayer aLayer) + { + if (this == MENTIONS_FROM_JSON) { + // Mention extraction only makes sense for span layers + return SpanLayerSupport.TYPE.equals(aLayer.getType()); + } + + return true; + } } diff --git a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/ExtractionModeSelect.java b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/ExtractionModeSelect.java index de82000e2c3..cdac9922f3b 100644 --- a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/ExtractionModeSelect.java +++ b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/ExtractionModeSelect.java @@ -19,32 +19,55 @@ import static java.util.Arrays.asList; +import java.util.Collections; + import org.apache.wicket.markup.html.form.DropDownChoice; import org.apache.wicket.markup.html.form.EnumChoiceRenderer; import org.apache.wicket.model.IModel; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; + public class ExtractionModeSelect extends DropDownChoice { private static final long serialVersionUID = 1789605828488016006L; - public ExtractionModeSelect(String aId) - { - super(aId); - } + private IModel recommender; - public ExtractionModeSelect(String aId, IModel aModel) + public ExtractionModeSelect(String aId, IModel aModel, + IModel aRecommender) { super(aId); setModel(aModel); + recommender = aRecommender; } @Override protected void onInitialize() { super.onInitialize(); - setChoiceRenderer(new EnumChoiceRenderer<>(this)); - setChoices(asList(ExtractionMode.values())); + } + + @Override + protected void onConfigure() + { + super.onConfigure(); + + if (!recommender.isPresent().getObject()) { + setChoices(Collections.emptyList()); + return; + } + + var validChoices = asList(ExtractionMode.values()).stream() // + .filter(e -> e.accepts(recommender.getObject().getLayer())) // + .toList(); + setChoices(validChoices); + + if (validChoices.size() == 1) { + setModelObject(validChoices.get(0)); + } + + setVisible(validChoices.size() > 1); } } diff --git a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommender.java b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommender.java index f67526eda4d..d356f463684 100644 --- a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommender.java +++ b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommender.java @@ -22,7 +22,6 @@ import java.io.IOException; import java.lang.invoke.MethodHandles; import java.nio.charset.Charset; -import java.util.List; import org.apache.commons.lang3.exception.ExceptionUtils; import org.apache.uima.cas.CAS; @@ -37,8 +36,8 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.NonTrainableRecommenderEngineImplBase; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; -import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.client.OllamaClient; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.client.OllamaGenerateRequest; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.prompt.PerAnnotationContextGenerator; @@ -47,7 +46,6 @@ import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.prompt.PromptContext; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.prompt.PromptContextGenerator; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.response.MentionsFromJsonExtractor; -import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.response.MentionsSample; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.response.ResponseAsLabelExtractor; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.response.ResponseExtractor; import de.tudarmstadt.ukp.inception.rendering.model.Range; @@ -88,12 +86,11 @@ public String getString(String aFullName, Charset aEncoding, } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { var responseExtractor = getResponseExtractor(); - List examples = responseExtractor.generate(this, aCas, - MAX_FEW_SHOT_EXAMPLES); + var examples = responseExtractor.generate(this, aCas, MAX_FEW_SHOT_EXAMPLES); getPromptContextGenerator().generate(this, aCas, aBegin, aEnd).forEach(promptContext -> { try { @@ -106,6 +103,8 @@ public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd responseExtractor.extract(this, aCas, promptContext, response); } catch (IOException e) { + aContext.error("Ollama [%s] failed to respond: %s", traits.getModel(), + ExceptionUtils.getRootCauseMessage(e)); LOG.error("Ollama [{}] failed to respond: {}", traits.getModel(), ExceptionUtils.getRootCauseMessage(e)); } diff --git a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderFactory.java b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderFactory.java index 76ee8019800..9ea98a4d9e8 100644 --- a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderFactory.java +++ b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderFactory.java @@ -17,7 +17,6 @@ */ package de.tudarmstadt.ukp.inception.recommendation.imls.ollama; -import static de.tudarmstadt.ukp.inception.support.WebAnnoConst.SPAN_TYPE; import static org.apache.uima.cas.CAS.TYPE_NAME_STRING; import java.lang.invoke.MethodHandles; @@ -34,12 +33,14 @@ import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; +import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanLayerSupport; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngineFactoryImplBase; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.client.OllamaClient; import de.tudarmstadt.ukp.inception.support.io.WatchedResourceFile; import de.tudarmstadt.ukp.inception.support.yaml.YamlUtil; +import de.tudarmstadt.ukp.inception.ui.core.docanno.layer.DocumentMetadataLayerSupport; public class OllamaRecommenderFactory extends RecommendationEngineFactoryImplBase @@ -87,7 +88,8 @@ public RecommendationEngine build(Recommender aRecommender) @Override public boolean accepts(AnnotationLayer aLayer, AnnotationFeature aFeature) { - return SPAN_TYPE.equals(aFeature.getLayer().getType()) + return (SpanLayerSupport.TYPE.equals(aFeature.getLayer().getType()) + || DocumentMetadataLayerSupport.TYPE.equals(aFeature.getLayer().getType())) && TYPE_NAME_STRING.equals(aFeature.getType()); } diff --git a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTraitsEditor.html b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTraitsEditor.html index 010b3773ffe..5f6080ca20b 100644 --- a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTraitsEditor.html +++ b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTraitsEditor.html @@ -17,6 +17,13 @@ --> + + +
@@ -32,7 +39,7 @@
- +
diff --git a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTraitsEditor.java b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTraitsEditor.java index 4b59dd142e9..fb153bb34e7 100644 --- a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTraitsEditor.java +++ b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTraitsEditor.java @@ -17,7 +17,13 @@ */ package de.tudarmstadt.ukp.inception.recommendation.imls.ollama; +import static de.tudarmstadt.ukp.inception.support.lambda.HtmlElementEvents.CHANGE_EVENT; +import static de.tudarmstadt.ukp.inception.support.wicket.WicketUtil.wrapInTryCatch; +import static java.lang.String.format; + +import java.io.IOException; import java.util.ArrayList; +import java.util.Collections; import java.util.List; import java.util.stream.Collectors; @@ -39,17 +45,26 @@ import org.apache.wicket.model.Model; import org.apache.wicket.model.PropertyModel; import org.apache.wicket.model.util.ListModel; +import org.apache.wicket.request.cycle.RequestCycle; import org.apache.wicket.spring.injection.annot.SpringBean; +import org.apache.wicket.validation.validator.UrlValidator; + +import com.googlecode.wicket.kendo.ui.KendoUIBehavior; +import com.googlecode.wicket.kendo.ui.form.combobox.ComboBox; +import com.googlecode.wicket.kendo.ui.form.combobox.ComboBoxBehavior; import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.AbstractTraitsEditor; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngineFactory; +import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.client.OllamaClientImpl; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.client.OllamaGenerateRequest; +import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.client.OllamaModel; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.client.Option; import de.tudarmstadt.ukp.inception.support.lambda.LambdaAjaxFormComponentUpdatingBehavior; import de.tudarmstadt.ukp.inception.support.lambda.LambdaAjaxLink; import de.tudarmstadt.ukp.inception.support.lambda.LambdaAjaxSubmitLink; +import de.tudarmstadt.ukp.inception.support.lambda.LambdaBehavior; import de.tudarmstadt.ukp.inception.support.markdown.MarkdownLabel; public class OllamaRecommenderTraitsEditor @@ -62,7 +77,7 @@ public class OllamaRecommenderTraitsEditor private @SpringBean RecommendationService recommendationService; private @SpringBean RecommendationEngineFactory toolFactory; - private final IModel traits; + private final CompoundPropertyModel traits; private final WebMarkupContainer optionSettingsContainer; private final IModel> optionSettings; @@ -93,22 +108,27 @@ protected void onSubmit() presetSelect.setModel(Model.of()); presetSelect.setChoiceRenderer(new ChoiceRenderer<>("name")); presetSelect.setChoices(aPresets); - presetSelect.add(new LambdaAjaxFormComponentUpdatingBehavior("change", _target -> { - var preset = presetSelect.getModelObject(); - if (preset != null) { - var settings = traits.getObject(); - settings.setPrompt(preset.getPrompt()); - settings.setExtractionMode(preset.getExtractionMode()); - settings.setFormat(preset.getFormat()); - settings.setPromptingMode(preset.getPromptingMode()); - settings.setRaw(preset.isRaw()); - } - _target.add(form); - })); + presetSelect.add(new LambdaAjaxFormComponentUpdatingBehavior(CHANGE_EVENT, + _target -> applyPreset(form, presetSelect.getModelObject(), _target))); form.add(presetSelect); - form.add(new TextField("url")); - form.add(new TextField("model")); + var modelsModel = LoadableDetachableModel.of(this::listModels); + var model = new ComboBox("model", modelsModel); + model.add(LambdaBehavior.onConfigure(() -> { + // Trigger a re-loading of the tagset from the server as constraints may have + // changed the ordering + modelsModel.detach(); + var target = RequestCycle.get().find(AjaxRequestTarget.class); + if (target.isPresent()) { + target.get().appendJavaScript(wrapInTryCatch(format( // + "var $w = %s; if ($w) { $w.dataSource.read(); }", + KendoUIBehavior.widget(this, ComboBoxBehavior.METHOD)))); + } + })); + model.setOutputMarkupId(true); + form.add(model); + form.add(new TextField("url").add(new LambdaAjaxFormComponentUpdatingBehavior( + CHANGE_EVENT, _target -> _target.add(model)))); form.add(new TextArea("prompt")); form.add(new CheckBox("raw").setOutputMarkupPlaceholderTag(true)); var markdownLabel = new MarkdownLabel("promptHints", @@ -116,10 +136,10 @@ protected void onSubmit() markdownLabel.setOutputMarkupId(true); form.add(markdownLabel); form.add(new PromptingModeSelect("promptingMode") - .add(new LambdaAjaxFormComponentUpdatingBehavior("change", _target -> { - _target.add(markdownLabel); - }))); - form.add(new ExtractionModeSelect("extractionMode")); + .add(new LambdaAjaxFormComponentUpdatingBehavior(CHANGE_EVENT, + _target -> _target.add(markdownLabel)))); + form.add(new ExtractionModeSelect("extractionMode", traits.bind("extractionMode"), + getModel())); form.add(new OllamaResponseFormatSelect("format")); add(form); @@ -144,6 +164,20 @@ protected void onSubmit() optionSettingsContainer.add(createOptionSettingsList("optionSettings", optionSettings)); } + private void applyPreset(Form aForm, Preset aPreset, + AjaxRequestTarget aTarget) + { + if (aPreset != null) { + var settings = traits.getObject(); + settings.setPrompt(aPreset.getPrompt()); + settings.setExtractionMode(aPreset.getExtractionMode()); + settings.setFormat(aPreset.getFormat()); + settings.setPromptingMode(aPreset.getPromptingMode()); + settings.setRaw(aPreset.isRaw()); + } + aTarget.add(aForm); + } + private ListView createOptionSettingsList(String aId, IModel> aOptionSettings) { @@ -183,4 +217,20 @@ private String getPromptHints() { return traits.getObject().getPromptingMode().getHints(); } + + private List listModels() + { + var url = traits.map(OllamaRecommenderTraits::getUrl).orElse(null).getObject(); + if (!new UrlValidator(new String[] { "http", "https" }).isValid(url)) { + return Collections.emptyList(); + } + + var client = new OllamaClientImpl(); + try { + return client.listModels(url).stream().map(OllamaModel::getName).toList(); + } + catch (IOException e) { + return Collections.emptyList(); + } + } } diff --git a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/client/OllamaClientImpl.java b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/client/OllamaClientImpl.java index 4aba17643ee..7dcf76b4fc3 100644 --- a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/client/OllamaClientImpl.java +++ b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/client/OllamaClientImpl.java @@ -156,8 +156,8 @@ public List listModels(String aUrl) throws IOException private void handleError(HttpResponse response) throws IOException { if (response.statusCode() >= HTTP_BAD_REQUEST) { - String responseBody = getResponseBody(response); - String msg = format("Request was not successful: [%d] - [%s]", response.statusCode(), + var responseBody = getResponseBody(response); + var msg = format("Request was not successful: [%d] - [%s]", response.statusCode(), responseBody); throw new IOException(msg); } diff --git a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/prompt/PromptingMode.java b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/prompt/PromptingMode.java index dc54330b650..414f5a3868c 100644 --- a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/prompt/PromptingMode.java +++ b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/prompt/PromptingMode.java @@ -24,6 +24,7 @@ public enum PromptingMode @JsonProperty("per-annotation") PER_ANNOTATION(""" Template variables: + * `text`: annotation text, * `sentence`: sentence containing annotation, * `examples`: labeled annotations"""), @@ -31,12 +32,14 @@ public enum PromptingMode @JsonProperty("per-sentence") PER_SENTENCE(""" Template variables: + * `text`: sentence text, * `examples`: labeled annotations"""), @JsonProperty("per-document") PER_DOCUMENT(""" Template variables: + * `text`: document text"""); private final String hints; diff --git a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/response/ResponseAsLabelExtractor.java b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/response/ResponseAsLabelExtractor.java index 369636cc8f8..9979c9e92f3 100644 --- a/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/response/ResponseAsLabelExtractor.java +++ b/inception/inception-imls-ollama/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/response/ResponseAsLabelExtractor.java @@ -48,12 +48,22 @@ public void extract(RecommendationEngine aEngine, CAS aCas, PromptContext aConte var predictedFeature = aEngine.getPredictedFeature(aCas); var isPredictionFeature = aEngine.getIsPredictionFeature(aCas); - var prediction = aCas.createAnnotation(predictedType, candidate.getBegin(), - candidate.getEnd()); - prediction.setFeatureValueFromString(predictedFeature, aResponse); - prediction.setBooleanValue(isPredictionFeature, true); - aCas.addFsToIndexes(prediction); + if (aCas.getAnnotationType().subsumes(predictedType)) { + var prediction = aCas.createAnnotation(predictedType, candidate.getBegin(), + candidate.getEnd()); + prediction.setFeatureValueFromString(predictedFeature, aResponse); + prediction.setBooleanValue(isPredictionFeature, true); + aCas.addFsToIndexes(prediction); - LOG.debug("Prediction generated [{}] -> [{}]", prediction.getCoveredText(), aResponse); + LOG.debug("Prediction generated: [{}] -> [{}]", prediction.getCoveredText(), aResponse); + } + else { + var prediction = aCas.createFS(predictedType); + prediction.setFeatureValueFromString(predictedFeature, aResponse); + prediction.setBooleanValue(isPredictionFeature, true); + aCas.addFsToIndexes(prediction); + + LOG.debug("Prediction generated: doc -> [{}]", aResponse); + } } } diff --git a/inception/inception-imls-ollama/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTest.java b/inception/inception-imls-ollama/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTest.java index ee8efa70574..85dd26c35ea 100644 --- a/inception/inception-imls-ollama/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTest.java +++ b/inception/inception-imls-ollama/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/ollama/OllamaRecommenderTest.java @@ -48,6 +48,7 @@ import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Token; import de.tudarmstadt.ukp.inception.recommendation.api.RecommenderTypeSystemUtils; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.recommendation.imls.ollama.client.OllamaClientImpl; import de.tudarmstadt.ukp.inception.support.test.http.HttpTestUtils; @@ -95,7 +96,7 @@ void testPerDocumentUsingReponseAsLabel() throws Exception traits.setExtractionMode(ExtractionMode.RESPONSE_AS_LABEL); var sut = new OllamaRecommender(recommender, traits, new OllamaClientImpl()); - sut.predict(new RecommenderContext(), cas); + sut.predict(new PredictionContext(new RecommenderContext()), cas); var predictions = cas.select(NamedEntity.class) .filter(ne -> getFeature(ne, FEATURE_NAME_IS_PREDICTION, Boolean.class)).toList(); @@ -120,7 +121,7 @@ void testPerDocumentUsingMentionsFromJsonList_Numbers() throws Exception traits.setExtractionMode(MENTIONS_FROM_JSON); var sut = new OllamaRecommender(recommender, traits, new OllamaClientImpl()); - sut.predict(new RecommenderContext(), cas); + sut.predict(new PredictionContext(new RecommenderContext()), cas); var predictions = cas.select(NamedEntity.class) .filter(ne -> getFeature(ne, FEATURE_NAME_IS_PREDICTION, Boolean.class)).toList(); @@ -143,7 +144,7 @@ void testPerDocumentUsingMentionsFromJsonList_Entities() throws Exception traits.setExtractionMode(MENTIONS_FROM_JSON); var sut = new OllamaRecommender(recommender, traits, new OllamaClientImpl()); - sut.predict(new RecommenderContext(), cas); + sut.predict(new PredictionContext(new RecommenderContext()), cas); var predictions = cas.select(NamedEntity.class) .filter(ne -> getFeature(ne, FEATURE_NAME_IS_PREDICTION, Boolean.class)).toList(); @@ -171,7 +172,7 @@ void testPerDocumentUsingMentionsFromJsonList_Politicians() throws Exception traits.setExtractionMode(MENTIONS_FROM_JSON); var sut = new OllamaRecommender(recommender, traits, new OllamaClientImpl()); - sut.predict(new RecommenderContext(), cas); + sut.predict(new PredictionContext(new RecommenderContext()), cas); var predictions = cas.select(NamedEntity.class) .filter(ne -> getFeature(ne, FEATURE_NAME_IS_PREDICTION, Boolean.class)).toList(); @@ -220,7 +221,7 @@ void testPerSentenceUsingMentionsFromJsonList_Politicians_fewShjot() throws Exce traits.setExtractionMode(MENTIONS_FROM_JSON); var sut = new OllamaRecommender(recommender, traits, new OllamaClientImpl()); - sut.predict(new RecommenderContext(), cas); + sut.predict(new PredictionContext(new RecommenderContext()), cas); var predictions = cas.select(NamedEntity.class) .filter(ne -> getFeature(ne, FEATURE_NAME_IS_PREDICTION, Boolean.class)).toList(); diff --git a/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatRecommender.java b/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatRecommender.java index 6ee353db93f..115b1180055 100644 --- a/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatRecommender.java +++ b/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatRecommender.java @@ -49,6 +49,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.LabelPair; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -137,7 +138,7 @@ public TrainingCapability getTrainingCapability() } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { var model = aContext.get(KEY_MODEL).orElseThrow( diff --git a/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/ner/OpenNlpNerRecommender.java b/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/ner/OpenNlpNerRecommender.java index 50d7a1a432a..54aa3c79ce2 100644 --- a/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/ner/OpenNlpNerRecommender.java +++ b/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/ner/OpenNlpNerRecommender.java @@ -43,6 +43,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.LabelPair; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -118,7 +119,7 @@ public TrainingCapability getTrainingCapability() } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { var model = aContext.get(KEY_MODEL).orElseThrow( diff --git a/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/pos/OpenNlpPosRecommender.java b/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/pos/OpenNlpPosRecommender.java index 23fe7bac5ce..7a67d4d6552 100644 --- a/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/pos/OpenNlpPosRecommender.java +++ b/inception/inception-imls-opennlp/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/pos/OpenNlpPosRecommender.java @@ -48,6 +48,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.LabelPair; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -119,7 +120,7 @@ public TrainingCapability getTrainingCapability() } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { var model = aContext.get(KEY_MODEL).orElseThrow( diff --git a/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatMetadataRecommenderTest.java b/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatMetadataRecommenderTest.java index 61b2e6bbc70..fdfc2ef5fb1 100644 --- a/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatMetadataRecommenderTest.java +++ b/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatMetadataRecommenderTest.java @@ -39,6 +39,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.RecommenderTypeSystemUtils; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.PercentageBasedSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.support.uima.SegmentationUtils; import de.tudarmstadt.ukp.inception.ui.core.docanno.layer.DocumentMetadataLayerSupport; @@ -107,7 +108,7 @@ public void thatPredictionWorks() throws Exception sut.train(context, casList); var predictionCas = makePredictionCas("I like cars.", feature); - sut.predict(context, predictionCas); + sut.predict(new PredictionContext(context), predictionCas); var predictions = getPredictionFSes(predictionCas, layer.getName()); diff --git a/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatRecommenderTest.java b/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatRecommenderTest.java index 79b43de5c00..0694e1952a4 100644 --- a/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatRecommenderTest.java +++ b/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/doccat/OpenNlpDoccatRecommenderTest.java @@ -53,6 +53,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.IncrementalSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.PercentageBasedSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.support.test.recommendation.DkproTestHelper; import de.tudarmstadt.ukp.inception.support.test.recommendation.RecommenderTestHelper; @@ -103,7 +104,7 @@ public void thatPredictionWorks() throws Exception sut.train(context, asList(cas)); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); var predictions = getPredictions(cas, NamedEntity.class); diff --git a/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/ner/OpenNlpNerRecommenderTest.java b/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/ner/OpenNlpNerRecommenderTest.java index 32da4e7f8f1..13f8e279a54 100644 --- a/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/ner/OpenNlpNerRecommenderTest.java +++ b/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/ner/OpenNlpNerRecommenderTest.java @@ -53,6 +53,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.IncrementalSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.PercentageBasedSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.support.test.recommendation.DkproTestHelper; import de.tudarmstadt.ukp.inception.support.test.recommendation.RecommenderTestHelper; @@ -103,7 +104,7 @@ public void thatPredictionWorks() throws Exception sut.train(context, asList(cas)); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); Collection predictions = JCasUtil.select(cas.getJCas(), NamedEntity.class); diff --git a/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/pos/OpenNlpPosRecommenderTest.java b/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/pos/OpenNlpPosRecommenderTest.java index 2c6f1d5a9f7..5d0c7a0ea9d 100644 --- a/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/pos/OpenNlpPosRecommenderTest.java +++ b/inception/inception-imls-opennlp/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/opennlp/pos/OpenNlpPosRecommenderTest.java @@ -47,6 +47,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.IncrementalSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.PercentageBasedSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.support.test.recommendation.DkproTestHelper; import de.tudarmstadt.ukp.inception.support.test.recommendation.RecommenderTestHelper; @@ -97,7 +98,7 @@ public void thatPredictionWorks() throws Exception sut.train(context, asList(cas)); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); List predictions = RecommenderTestHelper.getPredictions(cas, POS.class); diff --git a/inception/inception-imls-stringmatch/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/relation/StringMatchingRelationRecommender.java b/inception/inception-imls-stringmatch/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/relation/StringMatchingRelationRecommender.java index a3479b751b5..8fc35f0f5ba 100644 --- a/inception/inception-imls-stringmatch/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/relation/StringMatchingRelationRecommender.java +++ b/inception/inception-imls-stringmatch/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/relation/StringMatchingRelationRecommender.java @@ -50,6 +50,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.LabelPair; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -95,7 +96,7 @@ private MultiValuedMap, String> trainModel(List aDa } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { MultiValuedMap, String> model = aContext.get(KEY_MODEL).orElseThrow( diff --git a/inception/inception-imls-stringmatch/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/span/StringMatchingRecommender.java b/inception/inception-imls-stringmatch/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/span/StringMatchingRecommender.java index e2d569dea22..294a9b6d5e2 100644 --- a/inception/inception-imls-stringmatch/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/span/StringMatchingRecommender.java +++ b/inception/inception-imls-stringmatch/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/span/StringMatchingRecommender.java @@ -58,6 +58,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.LabelPair; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -198,7 +199,7 @@ public void train(RecommenderContext aContext, List aCasses) throws Recomme } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { Trie dict = aContext.get(KEY_MODEL).orElseThrow( diff --git a/inception/inception-imls-stringmatch/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/relation/StringMatchingRelationRecommenderTest.java b/inception/inception-imls-stringmatch/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/relation/StringMatchingRelationRecommenderTest.java index 26610cda11b..4e8f4d4b256 100644 --- a/inception/inception-imls-stringmatch/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/relation/StringMatchingRelationRecommenderTest.java +++ b/inception/inception-imls-stringmatch/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/relation/StringMatchingRelationRecommenderTest.java @@ -45,6 +45,7 @@ import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; import de.tudarmstadt.ukp.inception.annotation.storage.CasStorageSession; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.support.test.recommendation.RecommenderTestHelper; @@ -94,7 +95,7 @@ public void thatPredictingWorks() throws Exception CAS cas = loadSimpleCas(); sut.train(context, Collections.singletonList(cas)); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); List predictions = getPredictions(cas, RELATION_LAYER); diff --git a/inception/inception-imls-stringmatch/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/span/StringMatchingRecommenderTest.java b/inception/inception-imls-stringmatch/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/span/StringMatchingRecommenderTest.java index fa32d637b3c..14503e3df20 100644 --- a/inception/inception-imls-stringmatch/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/span/StringMatchingRecommenderTest.java +++ b/inception/inception-imls-stringmatch/src/test/java/de/tudarmstadt/ukp/inception/recommendation/imls/stringmatch/span/StringMatchingRecommenderTest.java @@ -63,6 +63,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.IncrementalSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.PercentageBasedSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; import de.tudarmstadt.ukp.inception.recommendation.imls.stringmatch.span.gazeteer.model.GazeteerEntry; import de.tudarmstadt.ukp.inception.support.test.recommendation.DkproTestHelper; @@ -116,7 +117,7 @@ public void thatPredictionWorks() throws Exception sut.train(context, Collections.singletonList(cas)); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); List predictions = getPredictions(cas, NamedEntity.class); @@ -141,7 +142,7 @@ public void thatPredictionForNoLabelAnnosWorks() throws Exception sut.train(context, Collections.singletonList(cas)); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); List predictions = getPredictions(cas, NamedEntity.class); @@ -170,7 +171,7 @@ public void thatPredictionForCharacterLevelLayerWorks() throws Exception sut.pretrain(gazeteer, context); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); List predictions = getPredictions(cas, NamedEntity.class); @@ -198,7 +199,7 @@ public void thatPredictionForCrossSentenceLayerWorks() throws Exception sut.pretrain(gazeteer, context); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); List predictions = getPredictions(cas, NamedEntity.class); @@ -242,7 +243,7 @@ public void thatPredictionWithPretrainigWorks() throws Exception sut.train(context, emptyList()); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); List predictions = getPredictions(cas, NamedEntity.class); @@ -275,7 +276,7 @@ public void thatPredictionWithPretrainigWorks_CaseInsensitve() throws Exception sut.train(context, emptyList()); - sut.predict(context, cas); + sut.predict(new PredictionContext(context), cas); List predictions = getPredictions(cas, NamedEntity.class); diff --git a/inception/inception-imls-weblicht/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/weblicht/WeblichtRecommender.java b/inception/inception-imls-weblicht/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/weblicht/WeblichtRecommender.java index 4c3d11a6b89..95550f9d6ab 100644 --- a/inception/inception-imls-weblicht/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/weblicht/WeblichtRecommender.java +++ b/inception/inception-imls-weblicht/src/main/java/de/tudarmstadt/ukp/inception/recommendation/imls/weblicht/WeblichtRecommender.java @@ -53,6 +53,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.DataSplitter; import de.tudarmstadt.ukp.inception.recommendation.api.evaluation.EvaluationResult; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -104,7 +105,7 @@ public void train(RecommenderContext aContext, List aCasses) } @Override - public Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException { // Begin and end are not used here because we do not know what kind of WebLicht pipeline diff --git a/inception/inception-js-api/src/main/ts/src/model/Annotation.ts b/inception/inception-js-api/src/main/ts/src/model/Annotation.ts index cf14b811642..6ff6c1712f0 100644 --- a/inception/inception-js-api/src/main/ts/src/model/Annotation.ts +++ b/inception/inception-js-api/src/main/ts/src/model/Annotation.ts @@ -39,6 +39,11 @@ export interface Annotation { */ score?: number + /** + * Whether to display the score in the UI or not. + */ + hideScore: boolean + /** * Comments */ diff --git a/inception/inception-js-api/src/main/ts/src/model/Relation.ts b/inception/inception-js-api/src/main/ts/src/model/Relation.ts index faa638495ff..2c11a8d9918 100644 --- a/inception/inception-js-api/src/main/ts/src/model/Relation.ts +++ b/inception/inception-js-api/src/main/ts/src/model/Relation.ts @@ -24,6 +24,7 @@ export class Relation implements Annotation { color?: string label?: string score?: number + hideScore: boolean comments?: Comment[] arguments: Array @@ -36,6 +37,7 @@ export class Relation implements Annotation { this.label = other.label this.comments = other.comments this.arguments = other.arguments + this.hideScore = other.hideScore } } } diff --git a/inception/inception-js-api/src/main/ts/src/model/Span.ts b/inception/inception-js-api/src/main/ts/src/model/Span.ts index a530429c6f7..fe7314cb705 100644 --- a/inception/inception-js-api/src/main/ts/src/model/Span.ts +++ b/inception/inception-js-api/src/main/ts/src/model/Span.ts @@ -28,6 +28,7 @@ export class Span implements Annotation { color?: string label?: string score?: number + hideScore: boolean comments?: Comment[] /** @@ -45,6 +46,7 @@ export class Span implements Annotation { this.label = other.label this.comments = other.comments this.clippingFlags = other.clippingFlags + this.hideScore = other.hideScore } } } diff --git a/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactAnnotationAttributes.ts b/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactAnnotationAttributes.ts index fc3583a932b..370f41fad85 100644 --- a/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactAnnotationAttributes.ts +++ b/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactAnnotationAttributes.ts @@ -37,4 +37,9 @@ export interface CompactAnnotationAttributes { * Score (optional) */ s: number + + /** + * Hide score (optional) + */ + hs: number } diff --git a/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactRelation.ts b/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactRelation.ts index 5adc31dd234..ada22e9236b 100644 --- a/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactRelation.ts +++ b/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactRelation.ts @@ -36,6 +36,7 @@ export function unpackCompactRelation (doc: AnnotatedText, raw: CompactRelation) cooked.color = raw[3]?.c cooked.label = raw[3]?.l cooked.score = raw[3]?.s + cooked.hideScore = raw[3]?.hs ? true : false cooked.comments = unpackCompactComments(doc, cooked, raw[3]?.cm) return cooked } diff --git a/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactSpan.ts b/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactSpan.ts index e7e1957383f..914f0cbac47 100644 --- a/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactSpan.ts +++ b/inception/inception-js-api/src/main/ts/src/model/compact_v2/CompactSpan.ts @@ -35,6 +35,7 @@ export function unpackCompactSpan (doc: AnnotatedText, raw: CompactSpan): Span { cooked.color = raw[3]?.c cooked.label = raw[3]?.l cooked.score = raw[3]?.s + cooked.hideScore = raw[3]?.hs ? true : false cooked.clippingFlags = raw[3]?.cl cooked.comments = unpackCompactComments(doc, cooked, raw[3]?.cm) return cooked diff --git a/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/config/DocumentMetadataLayerSupportAutoConfiguration.java b/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/config/DocumentMetadataLayerSupportAutoConfiguration.java index 9f4758831a4..dbbf7eea046 100644 --- a/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/config/DocumentMetadataLayerSupportAutoConfiguration.java +++ b/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/config/DocumentMetadataLayerSupportAutoConfiguration.java @@ -33,8 +33,8 @@ import de.tudarmstadt.ukp.inception.ui.core.docanno.event.DocumentMetadataAnnotationActionUndoSupport; import de.tudarmstadt.ukp.inception.ui.core.docanno.layer.DocumentMetadataLayerSingletonCreatingWatcher; import de.tudarmstadt.ukp.inception.ui.core.docanno.layer.DocumentMetadataLayerSupport; +import de.tudarmstadt.ukp.inception.ui.core.docanno.recommendation.MetadataSuggestionSupport; import de.tudarmstadt.ukp.inception.ui.core.docanno.sidebar.DocumentMetadataSidebarFactory; -import de.tudarmstadt.ukp.inception.ui.core.docanno.sidebar.MetadataSuggestionSupport; /** * Provides support for document-level annotations. diff --git a/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/layer/DocumentMetadataLayerSingletonCreatingWatcher.java b/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/layer/DocumentMetadataLayerSingletonCreatingWatcher.java index e77610f847e..8f3e48e8342 100644 --- a/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/layer/DocumentMetadataLayerSingletonCreatingWatcher.java +++ b/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/layer/DocumentMetadataLayerSingletonCreatingWatcher.java @@ -19,7 +19,6 @@ import java.io.IOException; import java.util.List; -import java.util.stream.Collectors; import org.apache.uima.cas.CAS; import org.springframework.context.event.EventListener; @@ -59,7 +58,7 @@ private List listMetadataLayers(Project aProject) return annotationService.listAnnotationLayer(aProject).stream() .filter(layer -> DocumentMetadataLayerSupport.TYPE.equals(layer.getType()) && layer.isEnabled()) - .collect(Collectors.toList()); + .toList(); } @EventListener @@ -72,13 +71,12 @@ public void onBeforeDocumentOpenedEvent(BeforeDocumentOpenedEvent aEvent) } CAS cas = aEvent.getCas(); - for (AnnotationLayer layer : listMetadataLayers(aEvent.getDocument().getProject())) { + for (var layer : listMetadataLayers(aEvent.getDocument().getProject())) { if (!getLayerSupport(layer).readTraits(layer).isSingleton()) { continue; } - DocumentMetadataLayerAdapter adapter = (DocumentMetadataLayerAdapter) annotationService - .getAdapter(layer); + var adapter = (DocumentMetadataLayerAdapter) annotationService.getAdapter(layer); if (cas.select(adapter.getAnnotationType(cas)).isEmpty()) { adapter.add(aEvent.getDocument(), aEvent.getSessionOwner(), cas); } diff --git a/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/sidebar/MetadataSuggestionSupport.java b/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/recommendation/MetadataSuggestionSupport.java similarity index 83% rename from inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/sidebar/MetadataSuggestionSupport.java rename to inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/recommendation/MetadataSuggestionSupport.java index 54e2830593b..84a418097f2 100644 --- a/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/sidebar/MetadataSuggestionSupport.java +++ b/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/recommendation/MetadataSuggestionSupport.java @@ -15,7 +15,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package de.tudarmstadt.ukp.inception.ui.core.docanno.sidebar; +package de.tudarmstadt.ukp.inception.ui.core.docanno.recommendation; import static de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion.FLAG_ALL; import static de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion.FLAG_OVERLAP; @@ -23,9 +23,11 @@ import static de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction.REJECTED; import java.lang.invoke.MethodHandles; +import java.util.ArrayList; import java.util.Collection; import java.util.List; import java.util.Objects; +import java.util.Optional; import org.apache.commons.lang3.NotImplementedException; import org.apache.uima.cas.AnnotationBaseFS; @@ -41,6 +43,7 @@ import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; import de.tudarmstadt.ukp.inception.recommendation.api.LearningRecordService; import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; +import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionRenderer; import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionSupport_ImplBase; import de.tudarmstadt.ukp.inception.recommendation.api.event.RecommendationRejectedEvent; import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; @@ -48,15 +51,18 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordChangeLocation; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction; import de.tudarmstadt.ukp.inception.recommendation.api.model.MetadataSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionGroup; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.ExtractionContext; import de.tudarmstadt.ukp.inception.schema.api.AnnotationSchemaService; import de.tudarmstadt.ukp.inception.schema.api.adapter.AnnotationException; import de.tudarmstadt.ukp.inception.schema.api.adapter.TypeAdapter; import de.tudarmstadt.ukp.inception.ui.core.docanno.layer.DocumentMetadataLayerAdapter; +import de.tudarmstadt.ukp.inception.ui.core.docanno.layer.DocumentMetadataLayerSupport; import de.tudarmstadt.ukp.inception.ui.core.docanno.layer.DocumentMetadataLayerTraits; public class MetadataSuggestionSupport - extends SuggestionSupport_ImplBase + extends SuggestionSupport_ImplBase { private static final Logger LOG = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass()); @@ -73,9 +79,18 @@ public MetadataSuggestionSupport(RecommendationService aRecommendationService, } @Override - public boolean accepts(AnnotationSuggestion aContext) + public boolean accepts(Recommender aContext) { - return aContext instanceof MetadataSuggestion; + if (!DocumentMetadataLayerSupport.TYPE.equals(aContext.getLayer().getType())) { + return false; + } + + var feature = aContext.getFeature(); + if (CAS.TYPE_NAME_STRING.equals(feature.getType()) || feature.isVirtualFeature()) { + return true; + } + + return false; } @Override @@ -269,4 +284,43 @@ var record = new LearningRecord(); record.setAnnotationFeature(aFeature); return record; } + + @Override + public Optional getRenderer() + { + return Optional.empty(); + } + + @Override + public List extractSuggestions(ExtractionContext ctx) + { + var result = new ArrayList(); + for (var predictedFS : ctx.getPredictionCas().select(ctx.getPredictedType())) { + if (!predictedFS.getBooleanValue(ctx.getPredictionFeature())) { + continue; + } + + var autoAcceptMode = getAutoAcceptMode(predictedFS, ctx.getModeFeature()); + var labels = getPredictedLabels(predictedFS, ctx.getLabelFeature(), + ctx.isMultiLabels()); + var score = predictedFS.getDoubleValue(ctx.getScoreFeature()); + var scoreExplanation = predictedFS.getStringValue(ctx.getScoreExplanationFeature()); + + for (var label : labels) { + var suggestion = MetadataSuggestion.builder() // + .withId(MetadataSuggestion.NEW_ID) // + .withGeneration(ctx.getGeneration()) // + .withRecommender(ctx.getRecommender()) // + .withDocument(ctx.getDocument()) // + .withLabel(label) // + .withUiLabel(label) // + .withScore(score) // + .withScoreExplanation(scoreExplanation) // + .withAutoAcceptMode(autoAcceptMode) // + .build(); + result.add(suggestion); + } + } + return result; + } } diff --git a/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/sidebar/DocumentMetadataAnnotationSelectionPanel.java b/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/sidebar/DocumentMetadataAnnotationSelectionPanel.java index 1b4ba3e0a45..fd0b616466b 100644 --- a/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/sidebar/DocumentMetadataAnnotationSelectionPanel.java +++ b/inception/inception-layer-docmetadata/src/main/java/de/tudarmstadt/ukp/inception/ui/core/docanno/sidebar/DocumentMetadataAnnotationSelectionPanel.java @@ -147,6 +147,7 @@ public DocumentMetadataAnnotationSelectionPanel(String aId, CasProvider aCasProv var availableLayers = LoadableDetachableModel.of(this::listCreatableMetadataLayers); var content = new WebMarkupContainer("content"); + content.add(visibleWhenNot(layers.map(List::isEmpty).orElse(true))); add(content); var layer = new DropDownChoice(CID_LAYER); @@ -164,11 +165,10 @@ public DocumentMetadataAnnotationSelectionPanel(String aId, CasProvider aCasProv layersContainer = new WebMarkupContainer(CID_LAYERS_CONTAINER); layersContainer.setOutputMarkupPlaceholderTag(true); layersContainer.add(createLayerGroupList()); - layersContainer.add(visibleWhenNot(availableLayers.map(List::isEmpty).orElse(true))); content.add(layersContainer); add(new WebMarkupContainer(CID_NO_LAYERS_WARNING) - .add(visibleWhen(availableLayers.map(List::isEmpty).orElse(true)))); + .add(visibleWhen(layers.map(List::isEmpty).orElse(true)))); } private void actionAccept(AjaxRequestTarget aTarget, AnnotationListItem aItem) @@ -314,8 +314,6 @@ private void actionSelect(AjaxRequestTarget aTarget, WebMarkupContainer containe private ListView createLayerGroupList() { - var availableLayers = listCreatableMetadataLayers(); - return new ListView(CID_LAYERS, layers) { private static final long serialVersionUID = 1L; @@ -331,7 +329,6 @@ protected void populateItem(ListItem aItem) var container = new WebMarkupContainer(CID_ANNOTATIONS_CONTAINER); container.setOutputMarkupPlaceholderTag(true); container.add(createAnnotationList(annotations)); - container.add(visibleWhen(() -> !availableLayers.isEmpty())); aItem.add(container); } @@ -521,8 +518,8 @@ private void generateSuggestionItems(AnnotationLayer aLayer, var featureSupport = fsRegistry.findExtension(feature).orElseThrow(); var annotation = featureSupport.renderFeatureValue(feature, suggestion.getLabel()); - items.add(new AnnotationListItem(suggestion.getVID(), annotation, aLayer, - singleton, suggestion.getScore())); + items.add(new AnnotationListItem(suggestion.getVID(), annotation, aLayer, false, + suggestion.getScore())); } } } diff --git a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/MetadataSuggestionExtractionTest.java b/inception/inception-layer-docmetadata/src/test/java/de/tudarmstadt/ukp/inception/ui/core/docanno/recommendation/MetadataSuggestionExtractionTest.java similarity index 78% rename from inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/MetadataSuggestionExtractionTest.java rename to inception/inception-layer-docmetadata/src/test/java/de/tudarmstadt/ukp/inception/ui/core/docanno/recommendation/MetadataSuggestionExtractionTest.java index be08bd861fe..6480b2a1c45 100644 --- a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/MetadataSuggestionExtractionTest.java +++ b/inception/inception-layer-docmetadata/src/test/java/de/tudarmstadt/ukp/inception/ui/core/docanno/recommendation/MetadataSuggestionExtractionTest.java @@ -15,7 +15,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package de.tudarmstadt.ukp.inception.recommendation.service; +package de.tudarmstadt.ukp.inception.ui.core.docanno.recommendation; import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_IS_PREDICTION; import static de.tudarmstadt.ukp.inception.support.uima.FeatureStructureBuilder.buildFS; @@ -30,19 +30,34 @@ import org.apache.uima.resource.metadata.TypeSystemDescription; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; +import org.springframework.context.ApplicationEventPublisher; import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; import de.tudarmstadt.ukp.clarin.webanno.model.Project; import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; +import de.tudarmstadt.ukp.inception.recommendation.api.LearningRecordService; +import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; import de.tudarmstadt.ukp.inception.recommendation.api.RecommenderTypeSystemUtils; import de.tudarmstadt.ukp.inception.recommendation.api.model.MetadataSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.ExtractionContext; +import de.tudarmstadt.ukp.inception.schema.api.AnnotationSchemaService; import de.tudarmstadt.ukp.inception.support.uima.SegmentationUtils; import de.tudarmstadt.ukp.inception.ui.core.docanno.layer.DocumentMetadataLayerSupport; +import de.tudarmstadt.ukp.inception.ui.core.docanno.recommendation.MetadataSuggestionSupport; +@ExtendWith(MockitoExtension.class) class MetadataSuggestionExtractionTest { + private @Mock RecommendationService recommendationService; + private @Mock LearningRecordService learningRecordService; + private @Mock ApplicationEventPublisher applicationEventPublisher; + private @Mock AnnotationSchemaService schemaService; + private Project project; private SourceDocument document; private TypeSystemDescription tsd; @@ -50,6 +65,8 @@ class MetadataSuggestionExtractionTest private TypeDescription metadataType; private FeatureDescription metadataLabelFeature; + private MetadataSuggestionSupport sut; + @BeforeEach void setup() throws Exception { @@ -73,6 +90,9 @@ void setup() throws Exception SegmentationUtils.splitSentences(originalCas); SegmentationUtils.tokenize(originalCas); + + sut = new MetadataSuggestionSupport(recommendationService, learningRecordService, + applicationEventPublisher, schemaService); } @Test @@ -103,8 +123,8 @@ void testDocumentMetadataExtraction() throws Exception .withFeature(FEATURE_NAME_IS_PREDICTION, true) // .buildAndAddToIndexes(); - var suggestions = SuggestionExtraction.extractSuggestions(1, originalCas, predictionCas, - document, recommender); + var ctx = new ExtractionContext(0, recommender, document, originalCas, predictionCas); + var suggestions = sut.extractSuggestions(ctx); assertThat(suggestions) // .filteredOn(a -> a instanceof MetadataSuggestion) // diff --git a/inception/inception-model-vdoc/src/main/java/de/tudarmstadt/ukp/inception/rendering/vmodel/VObject.java b/inception/inception-model-vdoc/src/main/java/de/tudarmstadt/ukp/inception/rendering/vmodel/VObject.java index 28320436fe4..f1543b0bf61 100644 --- a/inception/inception-model-vdoc/src/main/java/de/tudarmstadt/ukp/inception/rendering/vmodel/VObject.java +++ b/inception/inception-model-vdoc/src/main/java/de/tudarmstadt/ukp/inception/rendering/vmodel/VObject.java @@ -36,6 +36,7 @@ public abstract class VObject private String color; private String label; private double score; + private boolean hideScore; private boolean actionButtons; public VObject(AnnotationLayer aLayer, VID aVid, Map aFeatures) @@ -116,4 +117,14 @@ public void setScore(double aScore) { score = aScore; } + + public boolean isHideScore() + { + return hideScore; + } + + public void setHideScore(boolean aHideScore) + { + hideScore = aHideScore; + } } diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/RecommendationService.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/RecommendationService.java index 8c8d4c25f55..dcd73f818c1 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/RecommendationService.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/RecommendationService.java @@ -39,6 +39,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.model.RecommenderGeneralSettings; import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionDocumentGroup; import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionGroup; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngineFactory; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext; @@ -304,7 +305,9 @@ Predictions computePredictions(User aSessionOwner, Project aProject, * @param aLayer * the layer to which the suggestions belong. * @param aRecommendations - * the suggestions. + * the suggestions which must all be of the same type, e.g. all + * {@link SpanSuggestion}s. Use e.g. + * {@link SuggestionDocumentGroup#groupsOfType(Class, List)} to generate them. * @param aWindowBegin * the range of the document for which to update the suggestions. * @param aWindowEnd diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationTypeRenderer.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionRenderer.java similarity index 70% rename from inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationTypeRenderer.java rename to inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionRenderer.java index 0d4e51fa33c..2609c44dbb5 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationTypeRenderer.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionRenderer.java @@ -15,17 +15,18 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package de.tudarmstadt.ukp.inception.recommendation.render; +package de.tudarmstadt.ukp.inception.recommendation.api; -import de.tudarmstadt.ukp.inception.recommendation.api.model.Predictions; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; +import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionDocumentGroup; import de.tudarmstadt.ukp.inception.rendering.request.RenderRequest; import de.tudarmstadt.ukp.inception.rendering.vmodel.VDocument; -import de.tudarmstadt.ukp.inception.schema.api.adapter.TypeAdapter; /** * Type Adapters for span, arc, and chain annotations */ -public interface RecommendationTypeRenderer +public interface SuggestionRenderer { String COLOR = "#cccccc"; @@ -35,8 +36,12 @@ public interface RecommendationTypeRenderer * * @param aVdoc * a {@link VDocument} containing annotations for the given layer + * @param aSuggestions + * the suggestions to render * @param aRequest * a render request */ - void render(VDocument aVdoc, RenderRequest aRequest, Predictions aPredictions, T aAdapter); + void render(VDocument aVdoc, RenderRequest aRequest, + SuggestionDocumentGroup aSuggestions, + AnnotationLayer aLayer); } diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupport.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupport.java index 2a5aad69f2c..3d83e2ada1a 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupport.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupport.java @@ -18,6 +18,8 @@ package de.tudarmstadt.ukp.inception.recommendation.api; import java.util.Collection; +import java.util.List; +import java.util.Optional; import org.apache.uima.cas.AnnotationBaseFS; import org.apache.uima.cas.CAS; @@ -29,13 +31,15 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecord; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordChangeLocation; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionGroup; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.ExtractionContext; import de.tudarmstadt.ukp.inception.schema.api.adapter.AnnotationException; import de.tudarmstadt.ukp.inception.schema.api.adapter.TypeAdapter; import de.tudarmstadt.ukp.inception.support.extensionpoint.Extension; -public interface SuggestionSupport - extends Extension +public interface SuggestionSupport + extends Extension { AnnotationBaseFS acceptSuggestion(String aSessionOwner, SourceDocument aDocument, String aDataOwner, CAS aCas, TypeAdapter aAdapter, AnnotationFeature aFeature, @@ -58,4 +62,8 @@ void calculateSuggestionVisibility(String aSess LearningRecord toLearningRecord(SourceDocument aDocument, String aUsername, AnnotationSuggestion aSuggestion, AnnotationFeature aFeature, LearningRecordUserAction aUserAction, LearningRecordChangeLocation aLocation); + + Optional getRenderer(); + + List extractSuggestions(ExtractionContext aCtx); } diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupportRegistry.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupportRegistry.java index ecb85797fca..42f752a36f9 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupportRegistry.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupportRegistry.java @@ -17,11 +17,11 @@ */ package de.tudarmstadt.ukp.inception.recommendation.api; -import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.support.extensionpoint.ContextLookupExtensionPoint; public interface SuggestionSupportRegistry - extends ContextLookupExtensionPoint> + extends ContextLookupExtensionPoint { } diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupport_ImplBase.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupport_ImplBase.java index 03159fcc7d8..68436fc97c4 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupport_ImplBase.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/SuggestionSupport_ImplBase.java @@ -23,6 +23,9 @@ import org.apache.uima.cas.AnnotationBaseFS; import org.apache.uima.cas.CAS; +import org.apache.uima.cas.Feature; +import org.apache.uima.cas.FeatureStructure; +import org.apache.uima.fit.util.FSUtil; import org.springframework.beans.factory.BeanNameAware; import org.springframework.context.ApplicationEventPublisher; @@ -30,6 +33,7 @@ import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; import de.tudarmstadt.ukp.inception.recommendation.api.event.RecommendationAcceptedEvent; import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.model.AutoAcceptMode; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordChangeLocation; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction; import de.tudarmstadt.ukp.inception.schema.api.AnnotationSchemaService; @@ -37,8 +41,8 @@ import de.tudarmstadt.ukp.inception.schema.api.adapter.TypeAdapter; import de.tudarmstadt.ukp.inception.support.uima.ICasUtil; -public abstract class SuggestionSupport_ImplBase - implements SuggestionSupport, BeanNameAware +public abstract class SuggestionSupport_ImplBase + implements SuggestionSupport, BeanNameAware { protected final RecommendationService recommendationService; protected final LearningRecordService learningRecordService; @@ -95,4 +99,29 @@ protected void commmitLabel(String aSessionOwner, SourceDocument aDocument, Stri annotation, aFeature, aSuggestion.getLabel())); } } + + private static final String AUTO_ACCEPT_ON_FIRST_ACCESS = "on-first-access"; + + public static AutoAcceptMode getAutoAcceptMode(FeatureStructure aFS, Feature aModeFeature) + { + var autoAcceptMode = AutoAcceptMode.NEVER; + var autoAcceptFeatureValue = aFS.getStringValue(aModeFeature); + if (autoAcceptFeatureValue != null) { + switch (autoAcceptFeatureValue) { + case AUTO_ACCEPT_ON_FIRST_ACCESS: + autoAcceptMode = AutoAcceptMode.ON_FIRST_ACCESS; + } + } + return autoAcceptMode; + } + + public static String[] getPredictedLabels(FeatureStructure predictedFS, + Feature predictedFeature, boolean isStringMultiValue) + { + if (isStringMultiValue) { + return FSUtil.getFeature(predictedFS, predictedFeature, String[].class); + } + + return new String[] { predictedFS.getFeatureValueAsString(predictedFeature) }; + } } diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/AnnotationSuggestion.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/AnnotationSuggestion.java index e13dfb989c7..42cfc1c5946 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/AnnotationSuggestion.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/AnnotationSuggestion.java @@ -106,7 +106,7 @@ public AnnotationSuggestion(int aId, int aGeneration, int aAge, long aRecommende scoreExplanation = aScoreExplanation; recommenderId = aRecommenderId; documentName = aDocumentName; - autoAcceptMode = aAutoAcceptMode; + autoAcceptMode = aAutoAcceptMode != null ? aAutoAcceptMode : AutoAcceptMode.NEVER; hidingFlags = aHidingFlags; } diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/LearningRecord.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/LearningRecord.java index 7943e160046..b97e0c3a6a4 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/LearningRecord.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/LearningRecord.java @@ -50,6 +50,7 @@ public class LearningRecord { private static final long serialVersionUID = -8487663728083806672L; private static final int TOKEN_TEXT_LENGTH = 255; + private static final int LABEL_LENGTH = 255; @Id @GeneratedValue(strategy = GenerationType.IDENTITY) @@ -248,11 +249,16 @@ public void setSuggestionType(String aSuggestionType) suggestionType = aSuggestionType; } - public void setTokenText(String tokenText) + public void setTokenText(String aTokenText) { + if (aTokenText == null) { + tokenText = null; + return; + } + // Truncate the token text if it is too long - int targetLength = Math.min(tokenText.length(), TOKEN_TEXT_LENGTH); - this.tokenText = tokenText.substring(0, targetLength); + int targetLength = Math.min(aTokenText.length(), TOKEN_TEXT_LENGTH); + tokenText = aTokenText.substring(0, targetLength); } /** @@ -267,7 +273,14 @@ public String getAnnotation() public void setAnnotation(String aAnnotation) { - annotation = aAnnotation; + if (aAnnotation == null) { + annotation = null; + return; + } + + // Truncate the label if it is too long + int targetLength = Math.min(aAnnotation.length(), LABEL_LENGTH); + annotation = aAnnotation.substring(0, targetLength); } public LearningRecordUserAction getUserAction() diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/MetadataSuggestion.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/MetadataSuggestion.java index 7bc98f300cc..90219ba92df 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/MetadataSuggestion.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/MetadataSuggestion.java @@ -19,6 +19,8 @@ import java.io.Serializable; +import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; + public class MetadataSuggestion extends AnnotationSuggestion implements Serializable @@ -129,31 +131,42 @@ public Builder withRecommender(Recommender aRecommender) return this; } - public Builder withRecommenderId(long aRecommenderId) + @Deprecated + Builder withRecommenderId(long aRecommenderId) { this.recommenderId = aRecommenderId; return this; } - public Builder withRecommenderName(String aRecommenderName) + @Deprecated + Builder withRecommenderName(String aRecommenderName) { this.recommenderName = aRecommenderName; return this; } - public Builder withLayerId(long aLayerId) + @Deprecated + Builder withLayerId(long aLayerId) { this.layerId = aLayerId; return this; } - public Builder withFeature(String aFeature) + @Deprecated + Builder withFeature(String aFeature) { this.feature = aFeature; return this; } - public Builder withDocumentName(String aDocumentName) + public Builder withDocument(SourceDocument aDocument) + { + this.documentName = aDocument.getName(); + return this; + } + + @Deprecated + Builder withDocumentName(String aDocumentName) { this.documentName = aDocumentName; return this; diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/Predictions.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/Predictions.java index c725cb3a795..1a9b08f3a32 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/Predictions.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/Predictions.java @@ -157,7 +157,7 @@ public SuggestionDocumentGroup getGroupedPre Class type, String aDocumentName, AnnotationLayer aLayer, int aWindowBegin, int aWindowEnd) { - return new SuggestionDocumentGroup<>( + return SuggestionDocumentGroup.groupsOfType(type, getFlattenedPredictions(type, aDocumentName, aLayer, aWindowBegin, aWindowEnd)); } @@ -335,6 +335,11 @@ public List getPredictionsByRecommenderAndDocument( } } + public List getPredictionsByDocument(SourceDocument aDocument) + { + return getPredictionsByDocument(aDocument.getName()); + } + public List getPredictionsByDocument(String aDocumentName) { synchronized (predictionsLock) { @@ -345,6 +350,20 @@ public List getPredictionsByDocument(String aDocumentName) } } + public List getPredictionsByDocument(String aDocumentName, + int aWindowBegin, int aWindowEnd) + { + synchronized (predictionsLock) { + var byDocument = idxDocuments.getOrDefault(aDocumentName, emptyMap()); + return byDocument.entrySet().stream() // + .filter(f -> AnnotationPredicates.overlapping(f.getValue().getWindowBegin(), + f.getValue().getWindowEnd(), aWindowBegin, aWindowEnd)) + .sorted(comparingInt(e2 -> e2.getValue().getWindowBegin())) // + .map(Map.Entry::getValue) // + .collect(toList()); + } + } + public void markDocumentAsPredictionCompleted(SourceDocument aDocument) { synchronized (seenDocumentsForPrediction) { diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/RelationSuggestion.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/RelationSuggestion.java index ca1a7d22730..bf4602f917d 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/RelationSuggestion.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/RelationSuggestion.java @@ -21,6 +21,8 @@ import org.apache.commons.lang3.builder.ToStringBuilder; +import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; + public class RelationSuggestion extends AnnotationSuggestion implements Serializable @@ -39,21 +41,6 @@ private RelationSuggestion(Builder builder) this.position = builder.position; } - /** - * @deprecated Use builder instead - */ - @Deprecated - public RelationSuggestion(int aId, long aRecommenderId, String aRecommenderName, long aLayerId, - String aFeature, String aDocumentName, int aSourceBegin, int aSourceEnd, - int aTargetBegin, int aTargetEnd, String aLabel, String aUiLabel, double aScore, - String aScoreExplanation, AutoAcceptMode aAutoAcceptMode) - { - super(aId, 0, 0, aRecommenderId, aRecommenderName, aLayerId, aFeature, aDocumentName, - aLabel, aUiLabel, aScore, aScoreExplanation, aAutoAcceptMode, 0); - - position = new RelationPosition(aSourceBegin, aSourceEnd, aTargetBegin, aTargetEnd); - } - // Getter and setter @Override @@ -182,30 +169,41 @@ public Builder withRecommender(Recommender aRecommender) return this; } - public Builder withRecommenderId(long aRecommenderId) + @Deprecated + Builder withRecommenderId(long aRecommenderId) { this.recommenderId = aRecommenderId; return this; } - public Builder withRecommenderName(String aRecommenderName) + @Deprecated + Builder withRecommenderName(String aRecommenderName) { this.recommenderName = aRecommenderName; return this; } - public Builder withLayerId(long aLayerId) + @Deprecated + Builder withLayerId(long aLayerId) { this.layerId = aLayerId; return this; } - public Builder withFeature(String aFeature) + @Deprecated + Builder withFeature(String aFeature) { this.feature = aFeature; return this; } + public Builder withDocument(SourceDocument aDocument) + { + this.documentName = aDocument.getName(); + return this; + } + + @Deprecated public Builder withDocumentName(String aDocumentName) { this.documentName = aDocumentName; @@ -259,5 +257,4 @@ public RelationSuggestion build() return new RelationSuggestion(this); } } - } diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SpanSuggestion.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SpanSuggestion.java index d6041865c59..3d5ba6aae1e 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SpanSuggestion.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SpanSuggestion.java @@ -21,6 +21,8 @@ import org.apache.commons.lang3.builder.ToStringBuilder; +import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; + public class SpanSuggestion extends AnnotationSuggestion implements Serializable @@ -41,22 +43,6 @@ private SpanSuggestion(Builder builder) coveredText = builder.coveredText; } - /** - * @deprecated Use builder instead. - */ - @Deprecated - public SpanSuggestion(int aId, long aRecommenderId, String aRecommenderName, long aLayerId, - String aFeature, String aDocumentName, int aBegin, int aEnd, String aCoveredText, - String aLabel, String aUiLabel, double aScore, String aScoreExplanation, - AutoAcceptMode aAutoAcceptMode) - { - super(aId, 0, 0, aRecommenderId, aRecommenderName, aLayerId, aFeature, aDocumentName, - aLabel, aUiLabel, aScore, aScoreExplanation, aAutoAcceptMode, 0); - - position = new Offset(aBegin, aEnd); - coveredText = aCoveredText; - } - // Getter and setter public String getCoveredText() @@ -164,7 +150,7 @@ public static final class Builder private String scoreExplanation; private Offset position; private String coveredText; - private AutoAcceptMode autoAcceptMode; + private AutoAcceptMode autoAcceptMode = AutoAcceptMode.NEVER; private int hidingFlags; private Builder() @@ -198,31 +184,42 @@ public Builder withRecommender(Recommender aRecommender) return this; } - public Builder withRecommenderId(long aRecommenderId) + @Deprecated + Builder withRecommenderId(long aRecommenderId) { this.recommenderId = aRecommenderId; return this; } - public Builder withRecommenderName(String aRecommenderName) + @Deprecated + Builder withRecommenderName(String aRecommenderName) { this.recommenderName = aRecommenderName; return this; } - public Builder withLayerId(long aLayerId) + @Deprecated + Builder withLayerId(long aLayerId) { this.layerId = aLayerId; return this; } - public Builder withFeature(String aFeature) + @Deprecated + Builder withFeature(String aFeature) { this.feature = aFeature; return this; } - public Builder withDocumentName(String aDocumentName) + public Builder withDocument(SourceDocument aDocument) + { + this.documentName = aDocument.getName(); + return this; + } + + @Deprecated + Builder withDocumentName(String aDocumentName) { this.documentName = aDocumentName; return this; @@ -252,6 +249,12 @@ public Builder withScoreExplanation(String aScoreExplanation) return this; } + public Builder withPosition(int aBegin, int aEnd) + { + this.position = new Offset(aBegin, aEnd); + return this; + } + public Builder withPosition(Offset aPosition) { this.position = aPosition; diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SuggestionDocumentGroup.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SuggestionDocumentGroup.java index 012305260f0..e85afc0daaf 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SuggestionDocumentGroup.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SuggestionDocumentGroup.java @@ -22,6 +22,7 @@ import java.util.AbstractCollection; import java.util.ArrayList; import java.util.Collection; +import java.util.Collections; import java.util.Iterator; import java.util.LinkedHashMap; import java.util.List; @@ -42,14 +43,13 @@ public class SuggestionDocumentGroup private Collection> groups; private String documentName; - public SuggestionDocumentGroup() + /** + * Use {@ink #groupByType(List)} or {@link #groupsOfType(Class, List)} instead to ensure that + * never empty groups are created. + */ + private SuggestionDocumentGroup(List aSuggestions) { groups = new ArrayList<>(); - } - - public SuggestionDocumentGroup(List aSuggestions) - { - this(); addAll(SuggestionGroup.group(aSuggestions)); } @@ -64,7 +64,7 @@ public SuggestionDocumentGroup(List aSuggestions) */ @SuppressWarnings("unchecked") public static SuggestionDocumentGroup groupsOfType( - Class type, List aSuggestions) + Class type, List aSuggestions) { var filteredSuggestions = aSuggestions.stream().filter(type::isInstance) // .toList(); @@ -74,11 +74,16 @@ public static SuggestionDocumentGroup groups /** * @param aSuggestions * the list to retrieve suggestions from - * @return a SuggestionDocumentGroup where only suggestions of type V are added + * @return suggestions grouped by suggestion type. There will not be any empty groups in the + * result. */ - public static Map, SuggestionDocumentGroup> groupByType( - List aSuggestions) + public static Map, SuggestionDocumentGroup> // + groupByType(List aSuggestions) { + if (aSuggestions == null || aSuggestions.isEmpty()) { + return Collections.emptyMap(); + } + var groups = new LinkedHashMap, List>(); for (var suggestion : aSuggestions) { diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SuggestionGroup.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SuggestionGroup.java index da7526e937c..8390025f9ac 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SuggestionGroup.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SuggestionGroup.java @@ -178,13 +178,9 @@ public List bestSuggestions(Preferences aPreferences) // Determine the maximum score per Label Map maxScorePerLabel = new HashMap<>(); - for (LabelMapKey label : labelMap.keySet()) { - double maxScore = 0; - for (Entry classifier : labelMap.get(label).entrySet()) { - if (classifier.getValue().getScore() > maxScore) { - maxScore = classifier.getValue().getScore(); - } - } + for (var label : labelMap.keySet()) { + double maxScore = labelMap.get(label).values().stream() + .mapToDouble(AnnotationSuggestion::getScore).max().orElse(0.0d); maxScorePerLabel.put(label, maxScore); } @@ -404,6 +400,14 @@ public static SuggestionGroupCollector colle return new SuggestionGroupCollector(); } + /** + * @param + * type that all of the suggestions must have + * @param aSuggestions + * the suggestions + * @return suggestions grouped by {@code [layer, feature, position]}. There will be no empty + * groups. + */ public static Collection> group( Collection aSuggestions) { diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/AbstractTraitsEditor.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/AbstractTraitsEditor.java index d408fbc9432..1c7f25ee4f7 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/AbstractTraitsEditor.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/AbstractTraitsEditor.java @@ -17,7 +17,7 @@ */ package de.tudarmstadt.ukp.inception.recommendation.api.recommender; -import org.apache.wicket.markup.html.panel.Panel; +import org.apache.wicket.markup.html.panel.GenericPanel; import org.apache.wicket.model.IModel; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; @@ -40,7 +40,7 @@ * limitations under the License. */ public class AbstractTraitsEditor - extends Panel + extends GenericPanel { private static final long serialVersionUID = -5826029092354401342L; @@ -48,9 +48,4 @@ public AbstractTraitsEditor(String aId, IModel aRecommender) { super(aId, aRecommender); } - - public Recommender getModelObject() - { - return (Recommender) getDefaultModelObject(); - } } diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/ExtractionContext.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/ExtractionContext.java new file mode 100644 index 00000000000..6d333fc1889 --- /dev/null +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/ExtractionContext.java @@ -0,0 +1,165 @@ +/* + * Licensed to the Technische Universität Darmstadt under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The Technische Universität Darmstadt + * licenses this file to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. + * + * 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.tudarmstadt.ukp.inception.recommendation.api.recommender; + +import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_AUTO_ACCEPT_MODE_SUFFIX; +import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_IS_PREDICTION; +import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_SCORE_EXPLANATION_SUFFIX; +import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_SCORE_SUFFIX; +import static org.apache.uima.cas.CAS.TYPE_NAME_STRING_ARRAY; + +import org.apache.uima.cas.CAS; +import org.apache.uima.cas.Feature; +import org.apache.uima.cas.Type; +import org.apache.uima.fit.util.CasUtil; + +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; +import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; + +public final class ExtractionContext +{ + private final int generation; + + private final SourceDocument document; + private final CAS originalCas; + private final CAS predictionCas; + private final String documentText; + + private final Recommender recommender; + + private final AnnotationLayer layer; + private final String typeName; + private final String featureName; + + private final Type predictedType; + + private final Feature labelFeature; + private final Feature scoreFeature; + private final Feature scoreExplanationFeature; + private final Feature modeFeature; + private final Feature predictionFeature; + + private final boolean isMultiLabels; + + public ExtractionContext(int aGeneration, Recommender aRecommender, SourceDocument aDocument, + CAS aOriginalCas, CAS aPredictionCas) + { + recommender = aRecommender; + + document = aDocument; + originalCas = aOriginalCas; + documentText = originalCas.getDocumentText(); + predictionCas = aPredictionCas; + + generation = aGeneration; + layer = aRecommender.getLayer(); + featureName = aRecommender.getFeature().getName(); + typeName = layer.getName(); + + predictedType = CasUtil.getType(aPredictionCas, typeName); + labelFeature = predictedType.getFeatureByBaseName(featureName); + scoreFeature = predictedType.getFeatureByBaseName(featureName + FEATURE_NAME_SCORE_SUFFIX); + scoreExplanationFeature = predictedType + .getFeatureByBaseName(featureName + FEATURE_NAME_SCORE_EXPLANATION_SUFFIX); + modeFeature = predictedType + .getFeatureByBaseName(featureName + FEATURE_NAME_AUTO_ACCEPT_MODE_SUFFIX); + predictionFeature = predictedType.getFeatureByBaseName(FEATURE_NAME_IS_PREDICTION); + isMultiLabels = TYPE_NAME_STRING_ARRAY.equals(labelFeature.getRange().getName()); + } + + public int getGeneration() + { + return generation; + } + + public SourceDocument getDocument() + { + return document; + } + + public CAS getOriginalCas() + { + return originalCas; + } + + public CAS getPredictionCas() + { + return predictionCas; + } + + public String getDocumentText() + { + return documentText; + } + + public Recommender getRecommender() + { + return recommender; + } + + public AnnotationLayer getLayer() + { + return layer; + } + + public String getTypeName() + { + return typeName; + } + + public String getFeatureName() + { + return featureName; + } + + public Type getPredictedType() + { + return predictedType; + } + + public Feature getLabelFeature() + { + return labelFeature; + } + + public Feature getScoreFeature() + { + return scoreFeature; + } + + public Feature getScoreExplanationFeature() + { + return scoreExplanationFeature; + } + + public Feature getModeFeature() + { + return modeFeature; + } + + public Feature getPredictionFeature() + { + return predictionFeature; + } + + public boolean isMultiLabels() + { + return isMultiLabels; + } +} diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/PredictionContext.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/PredictionContext.java new file mode 100644 index 00000000000..2020df9a604 --- /dev/null +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/PredictionContext.java @@ -0,0 +1,107 @@ +/* + * Licensed to the Technische Universität Darmstadt under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The Technische Universität Darmstadt + * licenses this file to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. + * + * 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.tudarmstadt.ukp.inception.recommendation.api.recommender; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.Optional; + +import de.tudarmstadt.ukp.clarin.webanno.security.model.User; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommenderContext.Key; +import de.tudarmstadt.ukp.inception.support.logging.LogMessage; + +public class PredictionContext +{ + private RecommenderContext modelContext; + private List messages; + private boolean closed = false; + private Optional user; + + public PredictionContext(RecommenderContext aCtx) + { + modelContext = aCtx; + messages = new ArrayList<>(); + } + + synchronized public Optional get(Key aKey) + { + return modelContext.get(aKey); + } + + synchronized public void info(String aFormat, Object... aValues) + { + if (closed) { + throw new IllegalStateException("Adding data to a closed context is not permitted."); + } + + messages.add(LogMessage.info(this, aFormat, aValues)); + } + + synchronized public void warn(String aFormat, Object... aValues) + { + if (closed) { + throw new IllegalStateException("Adding data to a closed context is not permitted."); + } + + messages.add(LogMessage.warn(this, aFormat, aValues)); + } + + synchronized public void error(String aFormat, Object... aValues) + { + if (closed) { + throw new IllegalStateException("Adding data to a closed context is not permitted."); + } + + messages.add(LogMessage.error(this, aFormat, aValues)); + } + + public List getMessages() + { + return messages; + } + + /** + * Close the context. Further modifications to the context are not permitted. + */ + synchronized public void close() + { + if (!closed) { + closed = true; + messages = Collections.unmodifiableList(messages); + } + } + + /** + * @return whether the context is closed. + */ + synchronized public boolean isClosed() + { + return closed; + } + + public Optional getUser() + { + return user; + } + + public void setUser(User aUser) + { + user = Optional.ofNullable(aUser); + } +} diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommendationEngine.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommendationEngine.java index d2494c4362a..2fa08e2934b 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommendationEngine.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommendationEngine.java @@ -84,7 +84,7 @@ public abstract void train(RecommenderContext aContext, List aCasses) * @throws RecommendationException * if there was a problem during prediction */ - public void predict(RecommenderContext aContext, CAS aCas) throws RecommendationException + public void predict(PredictionContext aContext, CAS aCas) throws RecommendationException { predict(aContext, aCas, 0, aCas.getDocumentText().length()); } @@ -111,7 +111,7 @@ public void predict(RecommenderContext aContext, CAS aCas) throws Recommendation * @throws RecommendationException * if there was a problem during prediction */ - public abstract Range predict(RecommenderContext aContext, CAS aCas, int aBegin, int aEnd) + public abstract Range predict(PredictionContext aContext, CAS aCas, int aBegin, int aEnd) throws RecommendationException; /** diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommendationEngineFactory.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommendationEngineFactory.java index 141f2b14256..498812c5e22 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommendationEngineFactory.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommendationEngineFactory.java @@ -72,4 +72,16 @@ default String getExportModelName(Recommender aRecommender) T readTraits(Recommender aRecommender); void writeTraits(Recommender aRecommender, T aTraits); + + /** + * The task of a ranker is to provide ordered suggestions instead of scored suggestions. While a + * ranker will usually set the score property of a suggestion, that score should not be + * displayed prominently in the user interface. + * + * @return {@code true} if the recommender is a ranker + */ + default boolean isRanker(Recommender aRecommender) + { + return false; + } } diff --git a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommenderContext.java b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommenderContext.java index 115729ed561..db159e58c5e 100644 --- a/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommenderContext.java +++ b/inception/inception-recommendation-api/src/main/java/de/tudarmstadt/ukp/inception/recommendation/api/recommender/RecommenderContext.java @@ -24,8 +24,6 @@ import java.util.Map; import java.util.Optional; -import javax.annotation.Nullable; - import de.tudarmstadt.ukp.clarin.webanno.security.model.User; import de.tudarmstadt.ukp.inception.support.logging.LogMessage; @@ -43,7 +41,6 @@ public RecommenderContext() } @SuppressWarnings("unchecked") - @Nullable synchronized public Optional get(Key aKey) { return Optional.ofNullable((T) store.get(aKey.name)); diff --git a/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/AnnotationSuggestionTest.java b/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/AnnotationSuggestionTest.java index 296d2ee85d9..db4fb5cf2de 100644 --- a/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/AnnotationSuggestionTest.java +++ b/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/AnnotationSuggestionTest.java @@ -17,31 +17,44 @@ */ package de.tudarmstadt.ukp.inception.recommendation.api.model; -import static de.tudarmstadt.ukp.inception.recommendation.api.model.AutoAcceptMode.NEVER; import static java.util.Arrays.asList; import static org.assertj.core.api.Assertions.assertThat; -import java.util.List; - import org.junit.jupiter.api.Test; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; +import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; + public class AnnotationSuggestionTest { @Test public void thatEqualsAndHashCodeAndCompareToWorkCorrectly() { - SpanSuggestion rec1Sug1 = new SpanSuggestion(1, 1, "rec1", 1, "value", "doc1", 0, 1, "a", - "A", "#A", 0.1, "E1", NEVER); - SpanSuggestion rec1Sug2 = new SpanSuggestion(2, 1, "rec1", 1, "value", "doc1", 0, 1, "b", - "B", "#B", 0.2, "E2", NEVER); - SpanSuggestion rec2Sug1 = new SpanSuggestion(3, 2, "rec2", 1, "value", "doc1", 0, 1, "c", - "C", "#C", 0.1, "E1", NEVER); - SpanSuggestion rec2Sug2 = new SpanSuggestion(4, 2, "rec2", 1, "value", "doc1", 0, 1, "d", - "D", "#D", 0.3, "E3", NEVER); - - List all = asList(rec1Sug1, rec1Sug2, rec2Sug1, rec2Sug2); - for (SpanSuggestion x : all) { - for (SpanSuggestion y : all) { + var doc = SourceDocument.builder().withName("doc1").build(); + var layer = AnnotationLayer.builder().withId(1l).build(); + var feature = AnnotationFeature.builder().withLayer(layer).withName("value").build(); + var rec1 = Recommender.builder().withId(1l).withLayer(layer).withFeature(feature).build(); + var rec2 = Recommender.builder().withId(2l).withLayer(layer).withFeature(feature).build(); + + var builder = SpanSuggestion.builder().withLayerId(1).withFeature("value").withDocument(doc) + .withPosition(0, 1); + + builder.withRecommender(rec1); + var rec1Sug1 = builder.withId(1).withCoveredText("a").withLabel("A").withUiLabel("#A") + .withScore(0.1).withScoreExplanation("E1").build(); + var rec1Sug2 = builder.withId(2).withCoveredText("b").withLabel("B").withUiLabel("#B") + .withScore(0.2).withScoreExplanation("E2").build(); + + builder.withRecommender(rec2); + var rec2Sug1 = builder.withId(3).withCoveredText("c").withLabel("C").withUiLabel("#C") + .withScore(0.1).withScoreExplanation("E1").build(); + var rec2Sug2 = builder.withId(4).withCoveredText("d").withLabel("D").withUiLabel("#D") + .withScore(0.3).withScoreExplanation("E3").build(); + + var all = asList(rec1Sug1, rec1Sug2, rec2Sug1, rec2Sug2); + for (var x : all) { + for (var y : all) { if (x == y) { assertThat(x).isEqualTo(y); assertThat(y).isEqualTo(x); diff --git a/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/PredictionGroupTest.java b/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/PredictionGroupTest.java index 4dec3c7510b..67ed18f16b8 100644 --- a/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/PredictionGroupTest.java +++ b/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/PredictionGroupTest.java @@ -17,53 +17,82 @@ */ package de.tudarmstadt.ukp.inception.recommendation.api.model; -import static de.tudarmstadt.ukp.inception.recommendation.api.model.AutoAcceptMode.NEVER; import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.within; +import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; +import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; + public class PredictionGroupTest { + private SourceDocument doc; + private AnnotationLayer layer; + private AnnotationFeature feature; + private Recommender rec1; + private Recommender rec2; + + @BeforeEach + void setup() + { + doc = SourceDocument.builder().withId(123l).withName("doc1").build(); + layer = AnnotationLayer.builder().withId(1337l).withName("layer").build(); + feature = AnnotationFeature.builder().withId(1338l).withName("value").withLayer(layer) + .build(); + rec1 = Recommender.builder().withId(1l).withName("rec1").withLayer(layer) + .withFeature(feature).build(); + rec2 = Recommender.builder().withId(2l).withName("rec2").withLayer(layer) + .withFeature(feature).build(); + } + @Test public void thatAddingElementsToGroupWorks() { - var rec1Sug1 = new SpanSuggestion(1, 1, "rec1", 1, "value", "doc1", 0, 1, "a", "A", "#A", - 0.1, "E1", NEVER); - var rec1Sug2 = new SpanSuggestion(2, 1, "rec1", 1, "value", "doc1", 0, 1, "b", "B", "#B", - 0.2, "E2", NEVER); - var rec2Sug1 = new SpanSuggestion(3, 2, "rec2", 1, "value", "doc1", 0, 1, "c", "C", "#C", - 0.1, "E1", NEVER); - var rec2Sug2 = new SpanSuggestion(4, 2, "rec2", 1, "value", "doc1", 0, 1, "d", "D", "#D", - 0.3, "E3", NEVER); + var builder = SpanSuggestion.builder().withDocument(doc).withPosition(0, 1); + + builder.withRecommender(rec1); + var rec1Sug1 = builder.withId(1).withCoveredText("a").withLabel("A").withUiLabel("#A") + .withScore(0.1).withScoreExplanation("E1").build(); + var rec1Sug2 = builder.withId(2).withCoveredText("b").withLabel("B").withUiLabel("#B") + .withScore(0.2).withScoreExplanation("E2").build(); + + builder.withRecommender(rec2); + var rec2Sug1 = builder.withId(3).withCoveredText("c").withLabel("C").withUiLabel("#C") + .withScore(0.1).withScoreExplanation("E1").build(); + var rec2Sug2 = builder.withId(4).withCoveredText("d").withLabel("D").withUiLabel("#D") + .withScore(0.3).withScoreExplanation("E3").build(); // Ensure that group grows and that all elements are added properly var sut = new SuggestionGroup<>(); sut.add(rec1Sug1); - assertThat(sut).hasSize(1); - assertThat(sut).contains(rec1Sug1); + assertThat(sut).hasSize(1).contains(rec1Sug1); sut.add(rec1Sug2); - assertThat(sut).hasSize(2); - assertThat(sut).contains(rec1Sug2); + assertThat(sut).hasSize(2).contains(rec1Sug2); sut.add(rec2Sug1); - assertThat(sut).hasSize(3); - assertThat(sut).contains(rec2Sug1); + assertThat(sut).hasSize(3).contains(rec2Sug1); sut.add(rec2Sug2); - assertThat(sut).hasSize(4); - assertThat(sut).contains(rec2Sug2); + assertThat(sut).hasSize(4).contains(rec2Sug2); } @Test public void thatSortingWorks() { - var rec1Sug1 = new SpanSuggestion(1, 1, "rec1", 1, "value", "doc1", 0, 1, "a", "A", "#A", - 0.1, "E1", NEVER); - var rec1Sug2 = new SpanSuggestion(2, 1, "rec1", 1, "value", "doc1", 0, 1, "b", "B", "#B", - 0.2, "E2", NEVER); - var rec2Sug1 = new SpanSuggestion(3, 2, "rec2", 1, "value", "doc1", 0, 1, "c", "C", "#C", - 0.1, "E1", NEVER); - var rec2Sug2 = new SpanSuggestion(4, 2, "rec2", 1, "value", "doc1", 0, 1, "d", "D", "#D", - 0.3, "E3", NEVER); + var builder = SpanSuggestion.builder().withDocument(doc).withPosition(0, 1); + + builder.withRecommender(rec1); + var rec1Sug1 = builder.withId(1).withCoveredText("a").withLabel("A").withUiLabel("#A") + .withScore(0.1).withScoreExplanation("E1").build(); + var rec1Sug2 = builder.withId(2).withCoveredText("b").withLabel("B").withUiLabel("#B") + .withScore(0.2).withScoreExplanation("E2").build(); + + builder.withRecommender(rec2); + var rec2Sug1 = builder.withId(3).withCoveredText("c").withLabel("C").withUiLabel("#C") + .withScore(0.1).withScoreExplanation("E1").build(); + var rec2Sug2 = builder.withId(4).withCoveredText("d").withLabel("D").withUiLabel("#D") + .withScore(0.3).withScoreExplanation("E3").build(); var sut = new SuggestionGroup<>(rec1Sug1, rec1Sug2, rec2Sug1, rec2Sug2); @@ -83,14 +112,19 @@ public void thatSortingWorks() @Test public void thatTopDeltasAreCorrect() { - var rec1Sug1 = new SpanSuggestion(1, 1, "rec1", 1, "value", "doc1", 0, 1, "a", "A", "#A", - 0.1, "E1", NEVER); - var rec1Sug2 = new SpanSuggestion(2, 1, "rec1", 1, "value", "doc1", 0, 1, "b", "B", "#B", - 0.2, "E2", NEVER); - var rec2Sug1 = new SpanSuggestion(3, 2, "rec2", 1, "value", "doc1", 0, 1, "c", "C", "#C", - 0.1, "E1", NEVER); - var rec2Sug2 = new SpanSuggestion(4, 2, "rec2", 1, "value", "doc1", 0, 1, "d", "D", "#D", - 0.3, "E3", NEVER); + var builder = SpanSuggestion.builder().withDocument(doc).withPosition(0, 1); + + builder.withRecommender(rec1); + var rec1Sug1 = builder.withId(1).withCoveredText("a").withLabel("A").withUiLabel("#A") + .withScore(0.1).withScoreExplanation("E1").build(); + var rec1Sug2 = builder.withId(2).withCoveredText("b").withLabel("B").withUiLabel("#B") + .withScore(0.2).withScoreExplanation("E2").build(); + + builder.withRecommender(rec2); + var rec2Sug1 = builder.withId(3).withCoveredText("c").withLabel("C").withUiLabel("#C") + .withScore(0.1).withScoreExplanation("E1").build(); + var rec2Sug2 = builder.withId(4).withCoveredText("d").withLabel("D").withUiLabel("#D") + .withScore(0.3).withScoreExplanation("E3").build(); var sut = new SuggestionGroup<>(rec1Sug1, rec1Sug2, rec2Sug1, rec2Sug2); diff --git a/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/PredictionsTest.java b/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/PredictionsTest.java index c34b608b3aa..9dfa4621b10 100644 --- a/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/PredictionsTest.java +++ b/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/PredictionsTest.java @@ -37,8 +37,10 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; import de.tudarmstadt.ukp.clarin.webanno.model.Project; +import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; import de.tudarmstadt.ukp.clarin.webanno.security.model.User; import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Sentence; import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Token; @@ -112,12 +114,12 @@ void timeGetGroupedPredictions() throws Exception @Test void thatIdsAreAssigned() throws Exception { - var doc = "doc"; + var doc = SourceDocument.builder().withName("doc").build(); sut = new Predictions(user, user.getUsername(), project); sut.putPredictions(asList( // SpanSuggestion.builder() // .withId(AnnotationSuggestion.NEW_ID) // - .withDocumentName(doc) // + .withDocument(doc) // .build())); assertThat(sut.getPredictionsByDocument(doc)) // @@ -129,7 +131,7 @@ void thatIdsAreAssigned() throws Exception sut.putPredictions(asList( // SpanSuggestion.builder() // .withId(AnnotationSuggestion.NEW_ID) // - .withDocumentName(doc) // + .withDocument(doc) // .build())); sut.putPredictions(inheritedPredictions); @@ -148,25 +150,24 @@ private List generatePredictions(int aDocs, int aRecommend var tokens = cas.select(Token.class).asList(); var result = new ArrayList(); - for (int docId = 0; docId < aDocs; docId++) { - var docName = "doc" + docId; - for (var recId = 0; recId < aRecommenders; recId++) { - var recName = "rec" + recId; - var featName = "feat" + recId; + for (var docId = 0l; docId < aDocs; docId++) { + var doc = SourceDocument.builder().withId(docId).withName("doc" + docId).build(); + for (var recId = 0l; recId < aRecommenders; recId++) { + var feature = AnnotationFeature.builder().withId(recId).withName("feat" + recId) + .build(); + var rec = Recommender.builder().withId(recId).withName("rec" + recId) + .withLayer(layer).withFeature(feature).build(); for (int annId = 0; annId < aSuggestions; annId++) { var label = labels.get(rng.nextInt(labels.size())); var token = tokens.get(rng.nextInt(tokens.size())); var ann = SpanSuggestion.builder() // .withId(annId) // - .withDocumentName(docName) // - .withRecommenderName(recName) // - .withRecommenderId(recId) // - .withLayerId(layer.getId()) // + .withDocument(doc) // + .withRecommender(rec) // .withScore(rng.nextDouble()) // .withScoreExplanation(null) // .withLabel(label) // .withUiLabel(label) // - .withFeature(featName) // .withPosition(new Offset(token.getBegin(), token.getEnd())) // .withCoveredText(token.getCoveredText()) // .build(); diff --git a/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SpanSuggestionTest.java b/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SpanSuggestionTest.java index e21c1cb7ff5..bc8eeb1386b 100644 --- a/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SpanSuggestionTest.java +++ b/inception/inception-recommendation-api/src/test/java/de/tudarmstadt/ukp/inception/recommendation/api/model/SpanSuggestionTest.java @@ -21,25 +21,33 @@ import org.junit.jupiter.api.Test; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; +import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; + class SpanSuggestionTest { @Test void thatCloningRetainsAllInformation() { + var layer = AnnotationLayer.builder().withId(5l).withName("layer").build(); + var feature = AnnotationFeature.builder().withId(6l).withName("feature").withLayer(layer) + .build(); + var rec = Recommender.builder().withId(4l).withName("rec").withLayer(layer) + .withFeature(feature).build(); + var doc = SourceDocument.builder().withId(8l).withName("doc").build(); + var s = SpanSuggestion.builder() // .withId(1) // .withGeneration(2) // .withAge(3) // - .withRecommenderId(4) // - .withRecommenderName("rec") // - .withLayerId(5) // - .withFeature("feature") // - .withDocumentName("document") // + .withRecommender(rec) // + .withDocument(doc) // .withLabel("label") // .withUiLabel("uiLabel") // .withScore(6.0) // .withScoreExplanation("scoreExplanation") // - .withPosition(new Offset(1, 2)) // + .withPosition(1, 2) // .withCoveredText("coveredText") // .withAutoAcceptMode(AutoAcceptMode.ON_FIRST_ACCESS) // .withHidingFlags(7) // diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/RecommendationEditorExtension.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/RecommendationEditorExtension.java index 62c880a75df..60feb0d37f1 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/RecommendationEditorExtension.java +++ b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/RecommendationEditorExtension.java @@ -389,7 +389,7 @@ public List lookupLazyDetails(SourceDocument aDocument, User a var details = new VLazyDetailGroup(); for (var ao : sortedByScore) { var items = new ArrayList(); - if (ao.getScore() != -1) { + if (ao.getScore() > 0.0d) { items.add(String.format("Score: %.2f", ao.getScore())); } if (ao.getScoreExplanation().isPresent()) { diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/config/RecommenderServiceAutoConfiguration.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/config/RecommenderServiceAutoConfiguration.java index ab520c0e994..bb043fe68c0 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/config/RecommenderServiceAutoConfiguration.java +++ b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/config/RecommenderServiceAutoConfiguration.java @@ -51,15 +51,13 @@ import de.tudarmstadt.ukp.inception.recommendation.metrics.RecommendationMetricsImpl; import de.tudarmstadt.ukp.inception.recommendation.project.ProjectRecommendersMenuItem; import de.tudarmstadt.ukp.inception.recommendation.project.RecommenderProjectSettingsPanelFactory; -import de.tudarmstadt.ukp.inception.recommendation.render.RecommendationRelationRenderer; +import de.tudarmstadt.ukp.inception.recommendation.relation.RelationSuggestionSupport; import de.tudarmstadt.ukp.inception.recommendation.render.RecommendationRenderer; -import de.tudarmstadt.ukp.inception.recommendation.render.RecommendationSpanRenderer; import de.tudarmstadt.ukp.inception.recommendation.service.LayerRecommendtionSupportRegistryImpl; import de.tudarmstadt.ukp.inception.recommendation.service.RecommendationServiceImpl; import de.tudarmstadt.ukp.inception.recommendation.service.RecommenderFactoryRegistryImpl; -import de.tudarmstadt.ukp.inception.recommendation.service.RelationSuggestionSupport; -import de.tudarmstadt.ukp.inception.recommendation.service.SpanSuggestionSupport; import de.tudarmstadt.ukp.inception.recommendation.sidebar.RecommendationSidebarFactory; +import de.tudarmstadt.ukp.inception.recommendation.span.SpanSuggestionSupport; import de.tudarmstadt.ukp.inception.scheduling.SchedulingService; import de.tudarmstadt.ukp.inception.schema.api.AnnotationSchemaService; import de.tudarmstadt.ukp.inception.schema.api.feature.FeatureSupportRegistry; @@ -191,31 +189,11 @@ public RecommendationEventFooterItem recommendationEventFooterItem() @Bean public RecommendationRenderer recommendationRenderer(AnnotationSchemaService aAnnotationService, - RecommendationSpanRenderer aRecommendationSpanRenderer, - RecommendationRelationRenderer aRecommendationRelationRenderer, - RecommendationService aRecommendationService) - { - return new RecommendationRenderer(aAnnotationService, aRecommendationSpanRenderer, - aRecommendationRelationRenderer, aRecommendationService); - } - - @Bean - public RecommendationSpanRenderer recommendationSpanRenderer( - RecommendationService aRecommendationService, - AnnotationSchemaService aAnnotationService, FeatureSupportRegistry aFsRegistry, - RecommenderProperties aRecommenderProperties) - { - return new RecommendationSpanRenderer(aRecommendationService, aAnnotationService, - aFsRegistry, aRecommenderProperties); - } - - @Bean - public RecommendationRelationRenderer recommendationRelationRenderer( RecommendationService aRecommendationService, - AnnotationSchemaService aAnnotationService, FeatureSupportRegistry aFsRegistry) + SuggestionSupportRegistry aSuggestionSupportRegistry) { - return new RecommendationRelationRenderer(aRecommendationService, aAnnotationService, - aFsRegistry); + return new RecommendationRenderer(aAnnotationService, aRecommendationService, + aSuggestionSupportRegistry); } @Bean @@ -230,10 +208,12 @@ public SpanSuggestionSupport spanRecommendationSupport( RecommendationService aRecommendationService, LearningRecordService aLearningRecordService, ApplicationEventPublisher aApplicationEventPublisher, - AnnotationSchemaService aSchemaService) + AnnotationSchemaService aSchemaService, FeatureSupportRegistry aFeatureSupportRegistry, + RecommenderProperties aRecommenderProperties) { return new SpanSuggestionSupport(aRecommendationService, aLearningRecordService, - aApplicationEventPublisher, aSchemaService); + aApplicationEventPublisher, aSchemaService, aFeatureSupportRegistry, + aRecommenderProperties); } @Bean @@ -241,15 +221,15 @@ public RelationSuggestionSupport relationRecommendationSupport( RecommendationService aRecommendationService, LearningRecordService aLearningRecordService, ApplicationEventPublisher aApplicationEventPublisher, - AnnotationSchemaService aSchemaService) + AnnotationSchemaService aSchemaService, FeatureSupportRegistry aFeatureSupportRegistry) { return new RelationSuggestionSupport(aRecommendationService, aLearningRecordService, - aApplicationEventPublisher, aSchemaService); + aApplicationEventPublisher, aSchemaService, aFeatureSupportRegistry); } @Bean public SuggestionSupportRegistry layerRecommendtionSupportRegistry( - @Lazy @Autowired(required = false) List> aExtensions) + @Lazy @Autowired(required = false) List aExtensions) { return new LayerRecommendtionSupportRegistryImpl(aExtensions); } diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationRelationRenderer.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/relation/RelationSuggestionRenderer.java similarity index 70% rename from inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationRelationRenderer.java rename to inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/relation/RelationSuggestionRenderer.java index 63a329e13a9..9ac92151a12 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationRelationRenderer.java +++ b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/relation/RelationSuggestionRenderer.java @@ -15,24 +15,24 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package de.tudarmstadt.ukp.inception.recommendation.render; +package de.tudarmstadt.ukp.inception.recommendation.relation; -import static de.tudarmstadt.ukp.inception.annotation.storage.CasMetadataUtils.getSourceDocumentName; -import static de.tudarmstadt.ukp.inception.support.uima.WebAnnoCasUtil.getDocumentTitle; import static java.util.function.Function.identity; import static java.util.stream.Collectors.toMap; import static org.apache.uima.fit.util.CasUtil.selectAt; +import java.util.HashMap; import java.util.Map; import org.apache.uima.fit.util.CasUtil; import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; -import de.tudarmstadt.ukp.inception.annotation.layer.relation.RelationAdapter; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; -import de.tudarmstadt.ukp.inception.recommendation.api.model.Predictions; +import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionRenderer; +import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationSuggestion; -import de.tudarmstadt.ukp.inception.recommendation.config.RecommenderServiceAutoConfiguration; +import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionDocumentGroup; import de.tudarmstadt.ukp.inception.rendering.request.RenderRequest; import de.tudarmstadt.ukp.inception.rendering.vmodel.VArc; import de.tudarmstadt.ukp.inception.rendering.vmodel.VDocument; @@ -41,20 +41,14 @@ import de.tudarmstadt.ukp.inception.schema.api.feature.FeatureSupport; import de.tudarmstadt.ukp.inception.schema.api.feature.FeatureSupportRegistry; -/** - *

- * This class is exposed as a Spring Component via - * {@link RecommenderServiceAutoConfiguration#recommendationRelationRenderer}. - *

- */ -public class RecommendationRelationRenderer - implements RecommendationTypeRenderer +public class RelationSuggestionRenderer + implements SuggestionRenderer { private final RecommendationService recommendationService; private final AnnotationSchemaService annotationService; private final FeatureSupportRegistry fsRegistry; - public RecommendationRelationRenderer(RecommendationService aRecommendationService, + public RelationSuggestionRenderer(RecommendationService aRecommendationService, AnnotationSchemaService aAnnotationService, FeatureSupportRegistry aFsRegistry) { recommendationService = aRecommendationService; @@ -62,27 +56,15 @@ public RecommendationRelationRenderer(RecommendationService aRecommendationServi fsRegistry = aFsRegistry; } - /** - * Add annotations from the CAS, which is controlled by the window size, to the VDocument - * {@link VDocument} - * - * @param aVDoc - * A VDocument containing annotations for the given layer - * @param aPredictions - * the predictions to render - */ @Override - public void render(VDocument aVDoc, RenderRequest aRequest, Predictions aPredictions, - RelationAdapter aTypeAdapter) + public void render(VDocument aVDoc, RenderRequest aRequest, + SuggestionDocumentGroup aSuggestions, + AnnotationLayer aLayer) { var cas = aRequest.getCas(); - var layer = aTypeAdapter.getLayer(); // TODO #176 use the document Id once it it available in the CAS - var sourceDocumentName = getSourceDocumentName(cas).orElseGet(() -> getDocumentTitle(cas)); - var groupedPredictions = aPredictions.getGroupedPredictions(RelationSuggestion.class, - sourceDocumentName, layer, aRequest.getWindowBeginOffset(), - aRequest.getWindowEndOffset()); + var groupedPredictions = (SuggestionDocumentGroup) aSuggestions; // No recommendations to render for this layer if (groupedPredictions.isEmpty()) { @@ -91,21 +73,22 @@ public void render(VDocument aVDoc, RenderRequest aRequest, Predictions aPredict recommendationService.calculateSuggestionVisibility( aRequest.getSessionOwner().getUsername(), aRequest.getSourceDocument(), cas, - aRequest.getAnnotationUser().getUsername(), layer, groupedPredictions, + aRequest.getAnnotationUser().getUsername(), aLayer, groupedPredictions, aRequest.getWindowBeginOffset(), aRequest.getWindowEndOffset()); var pref = recommendationService.getPreferences(aRequest.getAnnotationUser(), - layer.getProject()); + aLayer.getProject()); - var attachType = CasUtil.getType(cas, layer.getAttachType().getName()); + var attachType = CasUtil.getType(cas, aLayer.getAttachType().getName()); // Bulk-load all the features of this layer to avoid having to do repeated DB accesses later - var features = annotationService.listSupportedFeatures(layer).stream() + var features = annotationService.listSupportedFeatures(aLayer).stream() .collect(toMap(AnnotationFeature::getName, identity())); + var rankerCache = new HashMap(); + for (var group : groupedPredictions) { for (var suggestion : group.bestSuggestions(pref)) { - // Skip rendering AnnotationObjects that should not be rendered if (!pref.isShowAllPredictions() && !suggestion.isVisible()) { continue; @@ -136,9 +119,19 @@ public void render(VDocument aVDoc, RenderRequest aRequest, Predictions aPredict ? Map.of(suggestion.getFeature(), annotation) : Map.of(); - var arc = new VArc(layer, suggestion.getVID(), VID.of(source), VID.of(target), + var isRanker = rankerCache.computeIfAbsent(suggestion.getRecommenderId(), id -> { + var recommender = recommendationService.getRecommender(id); + if (recommender != null) { + var factory = recommendationService.getRecommenderFactory(recommender); + return factory.map(f -> f.isRanker(recommender)).orElse(false); + } + return false; + }); + + var arc = new VArc(aLayer, suggestion.getVID(), VID.of(source), VID.of(target), "\uD83E\uDD16 " + suggestion.getUiLabel(), featureAnnotation, COLOR); arc.setScore(suggestion.getScore()); + arc.setHideScore(isRanker); aVDoc.add(arc); } diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/RelationSuggestionSupport.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/relation/RelationSuggestionSupport.java similarity index 77% rename from inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/RelationSuggestionSupport.java rename to inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/relation/RelationSuggestionSupport.java index 5b8239b2fc8..dde66836c6a 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/RelationSuggestionSupport.java +++ b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/relation/RelationSuggestionSupport.java @@ -15,7 +15,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package de.tudarmstadt.ukp.inception.recommendation.service; +package de.tudarmstadt.ukp.inception.recommendation.relation; import static de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion.FLAG_OVERLAP; import static de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion.FLAG_SKIPPED; @@ -32,12 +32,14 @@ import java.util.Collection; import java.util.Collections; import java.util.List; +import java.util.Optional; import org.apache.commons.collections4.multimap.ArrayListValuedHashMap; import org.apache.uima.cas.CAS; import org.apache.uima.cas.Type; import org.apache.uima.cas.text.AnnotationFS; import org.apache.uima.fit.util.CasUtil; +import org.apache.uima.jcas.tcas.Annotation; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.context.ApplicationEventPublisher; @@ -47,41 +49,59 @@ import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; import de.tudarmstadt.ukp.inception.annotation.layer.relation.RelationAdapter; +import de.tudarmstadt.ukp.inception.annotation.layer.relation.RelationLayerSupport; import de.tudarmstadt.ukp.inception.recommendation.api.LearningRecordService; import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; +import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionRenderer; import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionSupport_ImplBase; import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecord; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordChangeLocation; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction; import de.tudarmstadt.ukp.inception.recommendation.api.model.Position; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationPosition; import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionGroup; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.ExtractionContext; import de.tudarmstadt.ukp.inception.schema.api.AnnotationSchemaService; +import de.tudarmstadt.ukp.inception.schema.api.adapter.AnnotationComparisonUtils; import de.tudarmstadt.ukp.inception.schema.api.adapter.AnnotationException; import de.tudarmstadt.ukp.inception.schema.api.adapter.TypeAdapter; +import de.tudarmstadt.ukp.inception.schema.api.feature.FeatureSupportRegistry; public class RelationSuggestionSupport - extends SuggestionSupport_ImplBase + extends SuggestionSupport_ImplBase { private static final Logger LOG = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass()); public static final String TYPE = "RELATION"; + private final FeatureSupportRegistry featureSupportRegistry; + public RelationSuggestionSupport(RecommendationService aRecommendationService, LearningRecordService aLearningRecordService, ApplicationEventPublisher aApplicationEventPublisher, - AnnotationSchemaService aSchemaService) + AnnotationSchemaService aSchemaService, FeatureSupportRegistry aFeatureSupportRegistry) { super(aRecommendationService, aLearningRecordService, aApplicationEventPublisher, aSchemaService); + featureSupportRegistry = aFeatureSupportRegistry; } @Override - public boolean accepts(AnnotationSuggestion aContext) + public boolean accepts(Recommender aContext) { - return aContext instanceof RelationSuggestion; + if (!RelationLayerSupport.TYPE.equals(aContext.getLayer().getType())) { + return false; + } + + var feature = aContext.getFeature(); + if (CAS.TYPE_NAME_STRING.equals(feature.getType()) || feature.isVirtualFeature()) { + return true; + } + + return false; } /** @@ -359,4 +379,78 @@ var record = new LearningRecord(); record.setAnnotationFeature(aFeature); return record; } + + @Override + public Optional getRenderer() + { + return Optional.of(new RelationSuggestionRenderer(recommendationService, schemaService, + featureSupportRegistry)); + } + + @Override + public List extractSuggestions(ExtractionContext ctx) + { + // TODO Use adapter instead - once the method is no longer static + var sourceFeature = ctx.getPredictedType().getFeatureByBaseName(FEAT_REL_SOURCE); + var targetFeature = ctx.getPredictedType().getFeatureByBaseName(FEAT_REL_TARGET); + + var result = new ArrayList(); + for (var predictedFS : ctx.getPredictionCas().select(ctx.getPredictedType())) { + if (!predictedFS.getBooleanValue(ctx.getPredictionFeature())) { + continue; + } + + var source = (AnnotationFS) predictedFS.getFeatureValue(sourceFeature); + var target = (AnnotationFS) predictedFS.getFeatureValue(targetFeature); + + var originalSource = findEquivalentSpan(ctx.getOriginalCas(), source); + var originalTarget = findEquivalentSpan(ctx.getOriginalCas(), target); + if (originalSource.isEmpty() || originalTarget.isEmpty()) { + LOG.debug("Unable to find source or target of predicted relation in original CAS"); + continue; + } + + var autoAcceptMode = getAutoAcceptMode(predictedFS, ctx.getModeFeature()); + var labels = getPredictedLabels(predictedFS, ctx.getLabelFeature(), + ctx.isMultiLabels()); + var score = predictedFS.getDoubleValue(ctx.getScoreFeature()); + var scoreExplanation = predictedFS.getStringValue(ctx.getScoreExplanationFeature()); + var position = new RelationPosition(originalSource.get(), originalTarget.get()); + + for (var label : labels) { + var suggestion = RelationSuggestion.builder() // + .withId(RelationSuggestion.NEW_ID) // + .withGeneration(ctx.getGeneration()) // + .withRecommender(ctx.getRecommender()) // + .withDocumentName(ctx.getDocument().getName()) // + .withPosition(position) // + .withLabel(label) // + .withUiLabel(label) // + .withScore(score) // + .withScoreExplanation(scoreExplanation) // + .withAutoAcceptMode(autoAcceptMode) // + .build(); + result.add(suggestion); + } + } + return result; + } + + /** + * Locates an annotation in the given CAS which is equivalent of the provided annotation. + * + * @param aOriginalCas + * the original CAS. + * @param aAnnotation + * an annotation in the prediction CAS. return the equivalent in the original CAS. + */ + private static Optional findEquivalentSpan(CAS aOriginalCas, + AnnotationFS aAnnotation) + { + return aOriginalCas. select(aAnnotation.getType()) // + .at(aAnnotation) // + .filter(candidate -> AnnotationComparisonUtils.isEquivalentSpanAnnotation(candidate, + aAnnotation, null)) + .findFirst(); + } } diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationRenderer.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationRenderer.java index 1dbbe83c1f4..662cd589a2c 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationRenderer.java +++ b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationRenderer.java @@ -18,15 +18,21 @@ package de.tudarmstadt.ukp.inception.recommendation.render; import static de.tudarmstadt.ukp.clarin.webanno.model.Mode.ANNOTATION; -import static de.tudarmstadt.ukp.inception.support.WebAnnoConst.CHAIN_TYPE; +import static de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionDocumentGroup.groupByType; +import static java.util.function.Function.identity; +import static java.util.stream.Collectors.groupingBy; + +import java.util.HashMap; +import java.util.Optional; +import java.util.stream.Collectors; import org.springframework.core.annotation.Order; -import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Sentence; -import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Token; -import de.tudarmstadt.ukp.inception.annotation.layer.relation.RelationAdapter; -import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanAdapter; import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; +import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionSupport; +import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionSupportRegistry; +import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.config.RecommenderServiceAutoConfiguration; import de.tudarmstadt.ukp.inception.rendering.editorstate.AnnotatorState; import de.tudarmstadt.ukp.inception.rendering.pipeline.RenderStep; @@ -47,19 +53,16 @@ public class RecommendationRenderer public static final String ID = "RecommendationRenderer"; private final AnnotationSchemaService annotationService; - private final RecommendationSpanRenderer recommendationSpanRenderer; - private final RecommendationRelationRenderer recommendationRelationRenderer; private final RecommendationService recommendationService; + private final SuggestionSupportRegistry suggestionSupportRegistry; public RecommendationRenderer(AnnotationSchemaService aAnnotationService, - RecommendationSpanRenderer aRecommendationSpanRenderer, - RecommendationRelationRenderer aRecommendationRelationRenderer, - RecommendationService aRecommendationService) + RecommendationService aRecommendationService, + SuggestionSupportRegistry aSuggestionSupportRegistry) { annotationService = aAnnotationService; - recommendationSpanRenderer = aRecommendationSpanRenderer; - recommendationRelationRenderer = aRecommendationRelationRenderer; recommendationService = aRecommendationService; + suggestionSupportRegistry = aSuggestionSupportRegistry; } @Override @@ -100,24 +103,42 @@ public void render(VDocument aVDoc, RenderRequest aRequest) return; } - // Add the suggestions to the visual document + var suggestions = predictions.getPredictionsByDocument( + aRequest.getSourceDocument().getName(), aRequest.getWindowBeginOffset(), + aRequest.getWindowEndOffset()); + var suggestionsByLayer = suggestions.stream() + .collect(groupingBy(AnnotationSuggestion::getLayerId)); + + var recommenderCache = recommendationService.listRecommenders(aRequest.getProject()) + .stream().collect(Collectors.toMap(Recommender::getId, identity())); + var suggestionSupportCache = new HashMap>(); + for (var layer : aRequest.getVisibleLayers()) { - if (Token.class.getName().equals(layer.getName()) - || Sentence.class.getName().equals(layer.getName()) - || CHAIN_TYPE.equals(layer.getType()) - || !layer.isEnabled()) { /* Hide layer if not enabled */ + if (!layer.isEnabled() || layer.isReadonly()) { continue; } - var adapter = annotationService.getAdapter(layer); - if (adapter instanceof SpanAdapter) { - recommendationSpanRenderer.render(aVDoc, aRequest, predictions, - (SpanAdapter) adapter); + var suggestionsByType = groupByType(suggestionsByLayer.get(layer.getId())); + if (suggestionsByType.isEmpty()) { + continue; } - if (adapter instanceof RelationAdapter) { - recommendationRelationRenderer.render(aVDoc, aRequest, predictions, - (RelationAdapter) adapter); + for (var suggestionGroup : suggestionsByType.entrySet()) { + var suggestion = suggestionGroup.getValue().iterator().next().iterator().next(); + + var recommender = recommenderCache.get(suggestion.getRecommenderId()); + if (recommender == null) { + continue; + } + + var suggestionSupport = suggestionSupportCache.computeIfAbsent(recommender, + suggestionSupportRegistry::findGenericExtension); + if (suggestionSupport.isEmpty()) { + continue; + } + + suggestionSupport.get().getRenderer().ifPresent(renderer -> renderer.render(aVDoc, + aRequest, suggestionGroup.getValue(), layer)); } } } diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/LayerRecommendtionSupportRegistryImpl.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/LayerRecommendtionSupportRegistryImpl.java index 2c9ccd035d8..57742ff5de1 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/LayerRecommendtionSupportRegistryImpl.java +++ b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/LayerRecommendtionSupportRegistryImpl.java @@ -25,24 +25,23 @@ import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionSupport; import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionSupportRegistry; -import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.support.extensionpoint.ExtensionPoint_ImplBase; public class LayerRecommendtionSupportRegistryImpl - extends ExtensionPoint_ImplBase> + extends ExtensionPoint_ImplBase implements SuggestionSupportRegistry { @Autowired public LayerRecommendtionSupportRegistryImpl( - @Lazy @Autowired(required = false) List> aExtensions) + @Lazy @Autowired(required = false) List aExtensions) { super(aExtensions); } @SuppressWarnings("unchecked") @Override - public > Optional findGenericExtension( - AnnotationSuggestion aKey) + public Optional findGenericExtension(Recommender aKey) { return getExtensions().stream() // .filter(e -> e.accepts(aKey)) // diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImpl.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImpl.java index bf9fe770764..fa2f29b2f64 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImpl.java +++ b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImpl.java @@ -28,9 +28,9 @@ import static de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction.SKIPPED; import static de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionCapability.PREDICTION_USES_TEXT_ONLY; import static de.tudarmstadt.ukp.inception.recommendation.api.recommender.TrainingCapability.TRAINING_NOT_SUPPORTED; -import static de.tudarmstadt.ukp.inception.recommendation.service.SuggestionExtraction.extractSuggestions; import static de.tudarmstadt.ukp.inception.rendering.model.Range.rangeCoveringDocument; import static java.util.Collections.emptyList; +import static java.util.function.Function.identity; import static java.util.stream.Collectors.groupingBy; import static java.util.stream.Collectors.toList; import static java.util.stream.Collectors.toMap; @@ -40,6 +40,7 @@ import java.util.ArrayList; import java.util.Collection; import java.util.Collections; +import java.util.HashMap; import java.util.HashSet; import java.util.LinkedHashMap; import java.util.LinkedHashSet; @@ -51,6 +52,7 @@ import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; import java.util.concurrent.atomic.AtomicInteger; +import java.util.stream.Collectors; import org.apache.commons.collections4.MapIterator; import org.apache.commons.collections4.MultiValuedMap; @@ -92,7 +94,6 @@ import de.tudarmstadt.ukp.clarin.webanno.ui.annotation.AnnotationPage; import de.tudarmstadt.ukp.inception.annotation.events.AnnotationEvent; import de.tudarmstadt.ukp.inception.annotation.events.DocumentOpenedEvent; -import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanAdapter; import de.tudarmstadt.ukp.inception.annotation.storage.CasStorageSession; import de.tudarmstadt.ukp.inception.documents.api.DocumentService; import de.tudarmstadt.ukp.inception.documents.event.AfterCasWrittenEvent; @@ -107,6 +108,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; import de.tudarmstadt.ukp.inception.recommendation.api.RecommenderFactoryRegistry; import de.tudarmstadt.ukp.inception.recommendation.api.RecommenderTypeSystemUtils; +import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionSupport; import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionSupportRegistry; import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.AutoAcceptMode; @@ -122,6 +124,8 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionDocumentGroup; import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionGroup; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.ExtractionContext; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.PredictionContext; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngine; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngineFactory; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationException; @@ -177,7 +181,7 @@ public class RecommendationServiceImpl private final ProjectService projectService; private final ApplicationEventPublisher applicationEventPublisher; private final PreferencesService preferencesService; - private final SuggestionSupportRegistry layerRecommendtionSupportRegistry; + private final SuggestionSupportRegistry suggestionSupportRegistry; private final ConcurrentMap trainingTaskCounter; private final ConcurrentMap states; @@ -219,7 +223,7 @@ public RecommendationServiceImpl(PreferencesService aPreferencesService, projectService = aProjectService; entityManager = aEntityManager; applicationEventPublisher = aApplicationEventPublisher; - layerRecommendtionSupportRegistry = aLayerRecommendtionSupportRegistry; + suggestionSupportRegistry = aLayerRecommendtionSupportRegistry; trainingTaskCounter = new ConcurrentHashMap<>(); states = new ConcurrentHashMap<>(); @@ -601,6 +605,9 @@ private void autoAccept(AjaxRequestTarget aTarget, User aUser, SourceDocument aD } var page = (AnnotationPage) aTarget.getPage(); + if (!page.isEditable()) { + return; + } var predictions = getPredictions(aUser, aDocument.getProject()); if (predictions == null || predictions.isEmpty()) { @@ -608,10 +615,6 @@ private void autoAccept(AjaxRequestTarget aTarget, User aUser, SourceDocument aD return; } - if (!page.isEditable()) { - return; - } - CAS cas; try { cas = page.getEditorCas(); @@ -621,7 +624,10 @@ private void autoAccept(AjaxRequestTarget aTarget, User aUser, SourceDocument aD return; } - var count = 0; + var accepted = 0; + var recommenderCache = listRecommenders(aDocument.getProject()).stream() + .collect(Collectors.toMap(Recommender::getId, identity())); + var suggestionSupportCache = new HashMap>(); for (var prediction : predictions.getPredictionsByDocument(aDocument.getName())) { if (prediction.getAutoAcceptMode() != aAutoAcceptMode) { continue; @@ -634,30 +640,35 @@ private void autoAccept(AjaxRequestTarget aTarget, User aUser, SourceDocument aD prediction.clearAutoAccept(); } - var layer = schemaService.getLayer(prediction.getLayerId()); - var feature = schemaService.getFeature(prediction.getFeature(), layer); - var adapter = schemaService.getAdapter(layer); + var recommender = recommenderCache.get(prediction.getRecommenderId()); + if (recommender == null) { + continue; + } + + var suggestionSupport = suggestionSupportCache.computeIfAbsent(recommender, + suggestionSupportRegistry::findGenericExtension); + if (suggestionSupport.isEmpty()) { + continue; + } + + var feature = recommender.getFeature(); + var adapter = schemaService.getAdapter(recommender.getLayer()); adapter.silenceEvents(); try { - var rls = layerRecommendtionSupportRegistry.findGenericExtension(prediction); - if (rls.isPresent()) { - rls.get().acceptSuggestion(null, aDocument, aUser.getUsername(), cas, adapter, - feature, prediction, AUTO_ACCEPT, ACCEPTED); - rls.get().acceptSuggestion(null, aDocument, aUser.getUsername(), cas, - (SpanAdapter) adapter, feature, prediction, AUTO_ACCEPT, ACCEPTED); - count++; - } + suggestionSupport.get().acceptSuggestion(null, aDocument, aUser.getUsername(), cas, + adapter, feature, prediction, AUTO_ACCEPT, ACCEPTED); + accepted++; } catch (AnnotationException e) { LOG.debug("Not auto-accepting suggestion: {}", e.getMessage()); } } - predictions.log(LogMessage.info(this, "Auto-accepted [%d] suggestions", count)); - LOG.debug("Auto-accepted [{}] suggestions", count); + predictions.log(LogMessage.info(this, "Auto-accepted [%d] suggestions", accepted)); + LOG.debug("Auto-accepted [{}] suggestions", accepted); - if (count > 0) { + if (accepted > 0) { try { page.writeEditorCas(cas); } @@ -1041,8 +1052,8 @@ public AnnotationFS correctSuggestion(String aSessionOwner, SourceDocument aDocu var layer = schemaService.getLayer(aOriginalSuggestion.getLayerId()); var feature = schemaService.getFeature(aOriginalSuggestion.getFeature(), layer); - var originalRls = layerRecommendtionSupportRegistry - .findGenericExtension(aOriginalSuggestion); + var originalRls = suggestionSupportRegistry + .findGenericExtension(getRecommender(aOriginalSuggestion)); if (originalRls.isPresent()) { // If the action was a correction (i.e. suggestion label != annotation value) then // generate a rejection for the original value - we do not want the original value to @@ -1051,8 +1062,8 @@ public AnnotationFS correctSuggestion(String aSessionOwner, SourceDocument aDocu aLocation); } - var correctedRls = layerRecommendtionSupportRegistry - .findGenericExtension(aCorrectedSuggestion); + var correctedRls = suggestionSupportRegistry + .findGenericExtension(getRecommender(aCorrectedSuggestion)); if (correctedRls.isPresent()) { var adapter = schemaService.getAdapter(layer); @@ -1073,8 +1084,9 @@ public AnnotationBaseFS acceptSuggestion(String aSessionOwner, SourceDocument aD var layer = schemaService.getLayer(aSuggestion.getLayerId()); var feature = schemaService.getFeature(aSuggestion.getFeature(), layer); var adapter = schemaService.getAdapter(layer); + var recommender = getRecommender(aSuggestion); - var rls = layerRecommendtionSupportRegistry.findGenericExtension(aSuggestion); + var rls = suggestionSupportRegistry.findGenericExtension(recommender); if (rls.isPresent()) { return rls.get().acceptSuggestion(aSessionOwner, aDocument, aDataOwner, aCas, adapter, @@ -1450,9 +1462,6 @@ private void computePredictions(LazyCas aOriginalCas, return; } - RecommenderContext ctx = context.get(); - ctx.setUser(aSessionOwner); - Optional> maybeFactory = getRecommenderFactory(recommender); if (maybeFactory.isEmpty()) { @@ -1484,7 +1493,7 @@ private void computePredictions(LazyCas aOriginalCas, try { RecommendationEngine engine = factory.build(recommender); - if (!engine.isReadyForPrediction(ctx)) { + if (!engine.isReadyForPrediction(context.get())) { aPredictions.log(LogMessage.info(recommender.getName(), "Recommender context is not ready... skipping")); LOG.info("Recommender context {} for user {} in project {} is not ready for " // @@ -1513,8 +1522,11 @@ private void computePredictions(LazyCas aOriginalCas, engine.getRecommender(), activePredictions, aDocument, aSessionOwner); } else { + var ctx = new PredictionContext(context.get()); + ctx.setUser(aSessionOwner); generateSuggestions(aPredictions, ctx, engine, activePredictions, aDocument, originalCas, predictionCas, predictionBegin, predictionEnd); + ctx.getMessages().forEach(aPredictions::log); } } // Catching Throwable is intentional here as we want to continue the @@ -1715,7 +1727,7 @@ private void inheritSuggestionsAtDocumentLevel(Project aProject, SourceDocument /** * Invokes the engine to produce new suggestions. */ - void generateSuggestions(Predictions aIncomingPredictions, RecommenderContext aCtx, + void generateSuggestions(Predictions aIncomingPredictions, PredictionContext aCtx, RecommendationEngine aEngine, Predictions aActivePredictions, SourceDocument aDocument, CAS aOriginalCas, CAS aPredictionCas, int aPredictionBegin, int aPredictionEnd) throws RecommendationException @@ -1723,17 +1735,27 @@ void generateSuggestions(Predictions aIncomingPredictions, RecommenderContext aC var sessionOwner = aIncomingPredictions.getSessionOwner(); var recommender = aEngine.getRecommender(); + // Extract the suggestions from the data which the recommender has written into the CAS + var maybeSupportRegistry = suggestionSupportRegistry.findGenericExtension(recommender); + if (maybeSupportRegistry.isEmpty()) { + LOG.debug("There is no comparible suggestion support for {} - skipping prediction"); + aIncomingPredictions.log(LogMessage.warn(recommender.getName(), // + "Prediction skipped since there is no compatible suggestion support.")); + return; + } + // Perform the actual prediction aIncomingPredictions.log(LogMessage.info(recommender.getName(), "Generating predictions for layer [%s]...", recommender.getLayer().getUiName())); LOG.trace("{}[{}]: Generating predictions for layer [{}]", sessionOwner, recommender.getName(), recommender.getLayer().getUiName()); + var predictedRange = aEngine.predict(aCtx, aPredictionCas, aPredictionBegin, aPredictionEnd); - // Extract the suggestions from the data which the recommender has written into the CAS - var generatedSuggestions = extractSuggestions(aIncomingPredictions.getGeneration(), - aOriginalCas, aPredictionCas, aDocument, recommender); + var extractionContext = new ExtractionContext(aIncomingPredictions.getGeneration(), + recommender, aDocument, aOriginalCas, aPredictionCas); + var generatedSuggestions = maybeSupportRegistry.get().extractSuggestions(extractionContext); // Reconcile new suggestions with suggestions from previous run var reconciliationResult = reconcile(aActivePredictions, aDocument, recommender, @@ -1848,7 +1870,11 @@ public void calculateSuggestionVisibility(Strin return; } - var rls = layerRecommendtionSupportRegistry.findGenericExtension(maybeSuggestion.get()); + // All suggestions in the group must be of the same type. Even if they come from different + // recommenders, the recommenders must all be producing the same type of suggestion, so we + // can happily just take one of them in order to locate the suggestion support. + var recommender = getRecommender(maybeSuggestion.get()); + var rls = suggestionSupportRegistry.findGenericExtension(recommender); if (rls.isPresent()) { rls.get().calculateSuggestionVisibility(aSessionOwner, aDocument, aCas, aDataOwner, @@ -1994,7 +2020,7 @@ public void rejectSuggestion(String aSessionOwner, SourceDocument aDocument, Str AnnotationSuggestion suggestion, LearningRecordChangeLocation aAction) throws AnnotationException { - var rls = layerRecommendtionSupportRegistry.findGenericExtension(suggestion); + var rls = suggestionSupportRegistry.findGenericExtension(getRecommender(suggestion)); if (rls.isPresent()) { rls.get().rejectSuggestion(aSessionOwner, aDocument, aDataOwner, suggestion, aAction); @@ -2007,7 +2033,7 @@ public void skipSuggestion(String aSessionOwner, SourceDocument aDocument, Strin AnnotationSuggestion suggestion, LearningRecordChangeLocation aAction) throws AnnotationException { - var rls = layerRecommendtionSupportRegistry.findGenericExtension(suggestion); + var rls = suggestionSupportRegistry.findGenericExtension(getRecommender(suggestion)); if (rls.isPresent()) { rls.get().skipSuggestion(aSessionOwner, aDocument, aDataOwner, suggestion, aAction); @@ -2022,7 +2048,7 @@ public void logRecord(String aSessionOwner, SourceDocument aDocument, String aDa { LearningRecord record = null; - var rls = layerRecommendtionSupportRegistry.findGenericExtension(aSuggestion); + var rls = suggestionSupportRegistry.findGenericExtension(getRecommender(aSuggestion)); if (rls.isPresent()) { record = rls.get().toLearningRecord(aDocument, aDataOwner, aSuggestion, aFeature, diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/SuggestionExtraction.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/SuggestionExtraction.java deleted file mode 100644 index d71f4903997..00000000000 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/SuggestionExtraction.java +++ /dev/null @@ -1,444 +0,0 @@ -/* - * Licensed to the Technische Universität Darmstadt under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The Technische Universität Darmstadt - * licenses this file to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. - * - * 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.tudarmstadt.ukp.inception.recommendation.service; - -import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_AUTO_ACCEPT_MODE_SUFFIX; -import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_IS_PREDICTION; -import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_SCORE_EXPLANATION_SUFFIX; -import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_SCORE_SUFFIX; -import static de.tudarmstadt.ukp.inception.support.WebAnnoConst.FEAT_REL_SOURCE; -import static de.tudarmstadt.ukp.inception.support.WebAnnoConst.FEAT_REL_TARGET; -import static java.lang.Math.max; -import static java.lang.Math.min; -import static org.apache.uima.cas.CAS.TYPE_NAME_STRING_ARRAY; -import static org.apache.uima.fit.util.CasUtil.getType; - -import java.lang.invoke.MethodHandles; -import java.util.ArrayList; -import java.util.List; -import java.util.Optional; - -import org.apache.uima.cas.CAS; -import org.apache.uima.cas.Feature; -import org.apache.uima.cas.FeatureStructure; -import org.apache.uima.cas.Type; -import org.apache.uima.cas.text.AnnotationFS; -import org.apache.uima.fit.util.CasUtil; -import org.apache.uima.fit.util.FSUtil; -import org.apache.uima.jcas.cas.TOP; -import org.apache.uima.jcas.tcas.Annotation; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode; -import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; -import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; -import de.tudarmstadt.ukp.dkpro.core.api.segmentation.TrimUtils; -import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Sentence; -import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Token; -import de.tudarmstadt.ukp.inception.annotation.layer.relation.RelationLayerSupport; -import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanLayerSupport; -import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; -import de.tudarmstadt.ukp.inception.recommendation.api.model.AutoAcceptMode; -import de.tudarmstadt.ukp.inception.recommendation.api.model.MetadataSuggestion; -import de.tudarmstadt.ukp.inception.recommendation.api.model.Offset; -import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; -import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationPosition; -import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationSuggestion; -import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; -import de.tudarmstadt.ukp.inception.schema.api.adapter.AnnotationComparisonUtils; -import de.tudarmstadt.ukp.inception.ui.core.docanno.layer.DocumentMetadataLayerSupport; - -public class SuggestionExtraction -{ - private static final Logger LOG = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass()); - - private static final String AUTO_ACCEPT_ON_FIRST_ACCESS = "on-first-access"; - - static final class ExtractionContext - { - private final int generation; - - private final SourceDocument document; - private final CAS originalCas; - private final CAS predictionCas; - private final String documentText; - - private final Recommender recommender; - - private final AnnotationLayer layer; - private final String typeName; - private final String featureName; - - private final Type predictedType; - - private final Feature labelFeature; - private final Feature sourceFeature; - private final Feature targetFeature; - private final Feature scoreFeature; - private final Feature scoreExplanationFeature; - private final Feature modeFeature; - private final Feature predictionFeature; - - private final boolean isMultiLabels; - - private final List result; - - ExtractionContext(int aGeneration, Recommender aRecommender, SourceDocument aDocument, - CAS aOriginalCas, CAS aPredictionCas) - { - recommender = aRecommender; - - document = aDocument; - originalCas = aOriginalCas; - documentText = originalCas.getDocumentText(); - predictionCas = aPredictionCas; - - generation = aGeneration; - layer = aRecommender.getLayer(); - featureName = aRecommender.getFeature().getName(); - typeName = layer.getName(); - - predictedType = CasUtil.getType(aPredictionCas, typeName); - labelFeature = predictedType.getFeatureByBaseName(featureName); - sourceFeature = predictedType.getFeatureByBaseName(FEAT_REL_SOURCE); - targetFeature = predictedType.getFeatureByBaseName(FEAT_REL_TARGET); - scoreFeature = predictedType - .getFeatureByBaseName(featureName + FEATURE_NAME_SCORE_SUFFIX); - scoreExplanationFeature = predictedType - .getFeatureByBaseName(featureName + FEATURE_NAME_SCORE_EXPLANATION_SUFFIX); - modeFeature = predictedType - .getFeatureByBaseName(featureName + FEATURE_NAME_AUTO_ACCEPT_MODE_SUFFIX); - predictionFeature = predictedType.getFeatureByBaseName(FEATURE_NAME_IS_PREDICTION); - isMultiLabels = TYPE_NAME_STRING_ARRAY.equals(labelFeature.getRange().getName()); - - result = new ArrayList(); - } - } - - private static AutoAcceptMode getAutoAcceptMode(FeatureStructure aFS, Feature aModeFeature) - { - var autoAcceptMode = AutoAcceptMode.NEVER; - var autoAcceptFeatureValue = aFS.getStringValue(aModeFeature); - if (autoAcceptFeatureValue != null) { - switch (autoAcceptFeatureValue) { - case AUTO_ACCEPT_ON_FIRST_ACCESS: - autoAcceptMode = AutoAcceptMode.ON_FIRST_ACCESS; - } - } - return autoAcceptMode; - } - - private static String[] getPredictedLabels(FeatureStructure predictedFS, - Feature predictedFeature, boolean isStringMultiValue) - { - if (isStringMultiValue) { - return FSUtil.getFeature(predictedFS, predictedFeature, String[].class); - } - - return new String[] { predictedFS.getFeatureValueAsString(predictedFeature) }; - } - - static void extractDocumentMetadataSuggestion(ExtractionContext ctx, TOP predictedFS) - { - var autoAcceptMode = getAutoAcceptMode(predictedFS, ctx.modeFeature); - var labels = getPredictedLabels(predictedFS, ctx.labelFeature, ctx.isMultiLabels); - var score = predictedFS.getDoubleValue(ctx.scoreFeature); - var scoreExplanation = predictedFS.getStringValue(ctx.scoreExplanationFeature); - - for (var label : labels) { - var suggestion = MetadataSuggestion.builder() // - .withId(MetadataSuggestion.NEW_ID) // - .withGeneration(ctx.generation) // - .withRecommender(ctx.recommender) // - .withDocumentName(ctx.document.getName()) // - .withLabel(label) // - .withUiLabel(label) // - .withScore(score) // - .withScoreExplanation(scoreExplanation) // - .withAutoAcceptMode(autoAcceptMode) // - .build(); - ctx.result.add(suggestion); - } - } - - static void extractRelationSuggestion(ExtractionContext ctx, TOP predictedFS) - { - var autoAcceptMode = getAutoAcceptMode(predictedFS, ctx.modeFeature); - var labels = getPredictedLabels(predictedFS, ctx.labelFeature, ctx.isMultiLabels); - var score = predictedFS.getDoubleValue(ctx.scoreFeature); - var scoreExplanation = predictedFS.getStringValue(ctx.scoreExplanationFeature); - - var source = (AnnotationFS) predictedFS.getFeatureValue(ctx.sourceFeature); - var target = (AnnotationFS) predictedFS.getFeatureValue(ctx.targetFeature); - - var originalSource = findEquivalentSpan(ctx.originalCas, source); - var originalTarget = findEquivalentSpan(ctx.originalCas, target); - if (originalSource.isEmpty() || originalTarget.isEmpty()) { - LOG.debug("Unable to find source or target of predicted relation in original CAS"); - return; - } - - var position = new RelationPosition(originalSource.get(), originalTarget.get()); - - for (var label : labels) { - var suggestion = RelationSuggestion.builder() // - .withId(RelationSuggestion.NEW_ID) // - .withGeneration(ctx.generation) // - .withRecommender(ctx.recommender) // - .withDocumentName(ctx.document.getName()) // - .withPosition(position) // - .withLabel(label) // - .withUiLabel(label) // - .withScore(score) // - .withScoreExplanation(scoreExplanation) // - .withAutoAcceptMode(autoAcceptMode) // - .build(); - ctx.result.add(suggestion); - } - } - - static void extractSpanSuggestion(ExtractionContext ctx, TOP predictedFS) - { - var autoAcceptMode = getAutoAcceptMode(predictedFS, ctx.modeFeature); - var labels = getPredictedLabels(predictedFS, ctx.labelFeature, ctx.isMultiLabels); - var score = predictedFS.getDoubleValue(ctx.scoreFeature); - var scoreExplanation = predictedFS.getStringValue(ctx.scoreExplanationFeature); - - var predictedAnnotation = (Annotation) predictedFS; - var targetOffsets = getOffsets(ctx.layer.getAnchoringMode(), ctx.originalCas, - predictedAnnotation); - - if (targetOffsets.isEmpty()) { - LOG.debug("Prediction cannot be anchored to [{}]: {}", ctx.layer.getAnchoringMode(), - predictedAnnotation); - return; - } - - var offsets = targetOffsets.get(); - var coveredText = ctx.documentText.substring(offsets.getBegin(), offsets.getEnd()); - - for (var label : labels) { - var suggestion = SpanSuggestion.builder() // - .withId(RelationSuggestion.NEW_ID) // - .withGeneration(ctx.generation) // - .withRecommender(ctx.recommender) // - .withDocumentName(ctx.document.getName()) // - .withPosition(offsets) // - .withCoveredText(coveredText) // - .withLabel(label) // - .withUiLabel(label) // - .withScore(score) // - .withScoreExplanation(scoreExplanation) // - .withAutoAcceptMode(autoAcceptMode) // - .build(); - ctx.result.add(suggestion); - } - } - - static List extractSuggestions(int aGeneration, CAS aOriginalCas, - CAS aPredictionCas, SourceDocument aDocument, Recommender aRecommender) - { - var ctx = new ExtractionContext(aGeneration, aRecommender, aDocument, aOriginalCas, - aPredictionCas); - - for (var predictedFS : aPredictionCas.select(ctx.predictedType)) { - if (!predictedFS.getBooleanValue(ctx.predictionFeature)) { - continue; - } - - switch (ctx.layer.getType()) { - case SpanLayerSupport.TYPE: { - extractSpanSuggestion(ctx, predictedFS); - break; - } - case RelationLayerSupport.TYPE: { - extractRelationSuggestion(ctx, predictedFS); - break; - } - case DocumentMetadataLayerSupport.TYPE: { - extractDocumentMetadataSuggestion(ctx, predictedFS); - break; - } - default: - throw new IllegalStateException( - "Unsupported layer type [" + ctx.layer.getType() + "]"); - } - } - - return ctx.result; - } - - /** - * Calculates the offsets of the given predicted annotation in the original CAS . - * - * @param aMode - * the anchoring mode of the target layer - * @param aOriginalCas - * the original CAS. - * @param aPredictedAnnotation - * the predicted annotation. - * @return the proper offsets. - */ - static Optional getOffsets(AnchoringMode aMode, CAS aOriginalCas, - Annotation aPredictedAnnotation) - { - switch (aMode) { - case CHARACTERS: { - return getOffsetsAnchoredOnCharacters(aOriginalCas, aPredictedAnnotation); - } - case SINGLE_TOKEN: { - return getOffsetsAnchoredOnSingleTokens(aOriginalCas, aPredictedAnnotation); - } - case TOKENS: { - return getOffsetsAnchoredOnTokens(aOriginalCas, aPredictedAnnotation); - } - case SENTENCES: { - return getOffsetsAnchoredOnSentences(aOriginalCas, aPredictedAnnotation); - } - default: - throw new IllegalStateException("Unsupported anchoring mode: [" + aMode + "]"); - } - } - - private static Optional getOffsetsAnchoredOnCharacters(CAS aOriginalCas, - Annotation aPredictedAnnotation) - { - int[] offsets = { max(aPredictedAnnotation.getBegin(), 0), - min(aOriginalCas.getDocumentText().length(), aPredictedAnnotation.getEnd()) }; - TrimUtils.trim(aPredictedAnnotation.getCAS().getDocumentText(), offsets); - var begin = offsets[0]; - var end = offsets[1]; - return Optional.of(new Offset(begin, end)); - } - - private static Optional getOffsetsAnchoredOnSentences(CAS aOriginalCas, - Annotation aPredictedAnnotation) - { - var sentences = aOriginalCas.select(Sentence.class) // - .coveredBy(aPredictedAnnotation) // - .asList(); - - if (sentences.isEmpty()) { - // This can happen if a recommender uses different token boundaries (e.g. if a - // remote service performs its own tokenization). We might be smart here by - // looking for overlapping sentences instead of covered sentences. - LOG.trace("Discarding suggestion because no covered sentences were found: {}", - aPredictedAnnotation); - return Optional.empty(); - } - - var begin = sentences.get(0).getBegin(); - var end = sentences.get(sentences.size() - 1).getEnd(); - return Optional.of(new Offset(begin, end)); - } - - private static Optional getOffsetsAnchoredOnSingleTokens(CAS aOriginalCas, - Annotation aPredictedAnnotation) - { - Type tokenType = getType(aOriginalCas, Token.class); - var tokens = aOriginalCas. select(tokenType) // - .coveredBy(aPredictedAnnotation) // - .limit(2).asList(); - - if (tokens.isEmpty()) { - // This can happen if a recommender uses different token boundaries (e.g. if a - // remote service performs its own tokenization). We might be smart here by - // looking for overlapping tokens instead of contained tokens. - LOG.trace("Discarding suggestion because no covering token was found: {}", - aPredictedAnnotation); - return Optional.empty(); - } - - if (tokens.size() > 1) { - // We only want to accept single-token suggestions - LOG.trace("Discarding suggestion because only single-token suggestions are " - + "accepted: {}", aPredictedAnnotation); - return Optional.empty(); - } - - AnnotationFS token = tokens.get(0); - var begin = token.getBegin(); - var end = token.getEnd(); - return Optional.of(new Offset(begin, end)); - } - - static Optional getOffsetsAnchoredOnTokens(CAS aOriginalCas, - Annotation aPredictedAnnotation) - { - var tokens = aOriginalCas.select(Token.class) // - .coveredBy(aPredictedAnnotation) // - .asList(); - - if (tokens.isEmpty()) { - if (aPredictedAnnotation.getBegin() == aPredictedAnnotation.getEnd()) { - var pos = aPredictedAnnotation.getBegin(); - var allTokens = aOriginalCas.select(Token.class).asList(); - Token prevToken = null; - for (var token : allTokens) { - if (prevToken == null && pos < token.getBegin()) { - return Optional.of(new Offset(token.getBegin(), token.getBegin())); - } - - if (token.covering(aPredictedAnnotation)) { - return Optional.of(new Offset(pos, pos)); - } - - if (prevToken != null && pos < token.getBegin()) { - return Optional.of(new Offset(prevToken.getEnd(), prevToken.getEnd())); - } - - prevToken = token; - } - - if (prevToken != null && pos >= prevToken.getEnd()) { - return Optional.of(new Offset(prevToken.getEnd(), prevToken.getEnd())); - } - } - - // This can happen if a recommender uses different token boundaries (e.g. if a - // remote service performs its own tokenization). We might be smart here by - // looking for overlapping tokens instead of covered tokens. - LOG.trace("Discarding suggestion because no covered tokens were found: {}", - aPredictedAnnotation); - return Optional.empty(); - } - - var begin = tokens.get(0).getBegin(); - var end = tokens.get(tokens.size() - 1).getEnd(); - return Optional.of(new Offset(begin, end)); - } - - /** - * Locates an annotation in the given CAS which is equivalent of the provided annotation. - * - * @param aOriginalCas - * the original CAS. - * @param aAnnotation - * an annotation in the prediction CAS. return the equivalent in the original CAS. - */ - private static Optional findEquivalentSpan(CAS aOriginalCas, - AnnotationFS aAnnotation) - { - return aOriginalCas. select(aAnnotation.getType()) // - .at(aAnnotation) // - .filter(candidate -> AnnotationComparisonUtils.isEquivalentSpanAnnotation(candidate, - aAnnotation, null)) - .findFirst(); - } -} diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationSpanRenderer.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionRenderer.java similarity index 70% rename from inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationSpanRenderer.java rename to inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionRenderer.java index 264f3be32b4..ee5a066ed54 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/render/RecommendationSpanRenderer.java +++ b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionRenderer.java @@ -15,20 +15,22 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package de.tudarmstadt.ukp.inception.recommendation.render; +package de.tudarmstadt.ukp.inception.recommendation.span; import static java.util.function.Function.identity; import static java.util.stream.Collectors.toMap; +import java.util.HashMap; import java.util.Map; import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; -import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanAdapter; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; -import de.tudarmstadt.ukp.inception.recommendation.api.model.Predictions; +import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionRenderer; +import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionDocumentGroup; import de.tudarmstadt.ukp.inception.recommendation.config.RecommenderProperties; -import de.tudarmstadt.ukp.inception.recommendation.config.RecommenderServiceAutoConfiguration; import de.tudarmstadt.ukp.inception.rendering.request.RenderRequest; import de.tudarmstadt.ukp.inception.rendering.vmodel.VDocument; import de.tudarmstadt.ukp.inception.rendering.vmodel.VRange; @@ -36,21 +38,15 @@ import de.tudarmstadt.ukp.inception.schema.api.AnnotationSchemaService; import de.tudarmstadt.ukp.inception.schema.api.feature.FeatureSupportRegistry; -/** - *

- * This class is exposed as a Spring Component via - * {@link RecommenderServiceAutoConfiguration#recommendationSpanRenderer}. - *

- */ -public class RecommendationSpanRenderer - implements RecommendationTypeRenderer +public class SpanSuggestionRenderer + implements SuggestionRenderer { private final RecommendationService recommendationService; private final AnnotationSchemaService annotationService; private final FeatureSupportRegistry fsRegistry; private final RecommenderProperties recommenderProperties; - public RecommendationSpanRenderer(RecommendationService aRecommendationService, + public SpanSuggestionRenderer(RecommendationService aRecommendationService, AnnotationSchemaService aAnnotationService, FeatureSupportRegistry aFsRegistry, RecommenderProperties aRecommenderProperties) { @@ -60,42 +56,34 @@ public RecommendationSpanRenderer(RecommendationService aRecommendationService, recommenderProperties = aRecommenderProperties; } - /** - * Add annotations from the CAS, which is controlled by the window size, to the VDocument - * {@link VDocument} - * - * @param vdoc - * A VDocument containing annotations for the given layer - * @param aPredictions - * the predictions to render - */ @Override - public void render(VDocument vdoc, RenderRequest aRequest, Predictions aPredictions, - SpanAdapter aTypeAdapter) + public void render(VDocument vdoc, RenderRequest aRequest, + SuggestionDocumentGroup aSuggestions, + AnnotationLayer aLayer) { - var cas = aRequest.getCas(); - var layer = aTypeAdapter.getLayer(); - var groups = aPredictions.getGroupedPredictions(SpanSuggestion.class, - aRequest.getSourceDocument().getName(), layer, aRequest.getWindowBeginOffset(), - aRequest.getWindowEndOffset()); + var groups = (SuggestionDocumentGroup) aSuggestions; // No recommendations to render for this layer if (groups.isEmpty()) { return; } + var cas = aRequest.getCas(); + recommendationService.calculateSuggestionVisibility( aRequest.getSessionOwner().getUsername(), aRequest.getSourceDocument(), cas, - aRequest.getAnnotationUser().getUsername(), layer, groups, + aRequest.getAnnotationUser().getUsername(), aLayer, groups, aRequest.getWindowBeginOffset(), aRequest.getWindowEndOffset()); var pref = recommendationService.getPreferences(aRequest.getAnnotationUser(), - layer.getProject()); + aLayer.getProject()); // Bulk-load all the features of this layer to avoid having to do repeated DB accesses later - var features = annotationService.listSupportedFeatures(layer).stream() + var features = annotationService.listSupportedFeatures(aLayer).stream() .collect(toMap(AnnotationFeature::getName, identity())); + var rankerCache = new HashMap(); + for (var suggestionGroup : groups) { // Render annotations for each label for (var suggestion : suggestionGroup.bestSuggestions(pref)) { @@ -114,9 +102,19 @@ public void render(VDocument vdoc, RenderRequest aRequest, Predictions aPredicti ? Map.of(suggestion.getFeature(), annotation) : Map.of(); - var v = new VSpan(layer, suggestion.getVID(), range.get(), featureAnnotation, + var isRanker = rankerCache.computeIfAbsent(suggestion.getRecommenderId(), id -> { + var recommender = recommendationService.getRecommender(id); + if (recommender != null) { + var factory = recommendationService.getRecommenderFactory(recommender); + return factory.map(f -> f.isRanker(recommender)).orElse(false); + } + return false; + }); + + var v = new VSpan(aLayer, suggestion.getVID(), range.get(), featureAnnotation, COLOR); v.setScore(suggestion.getScore()); + v.setHideScore(isRanker); v.setActionButtons(recommenderProperties.isActionButtonsEnabled()); vdoc.add(v); diff --git a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/SpanSuggestionSupport.java b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionSupport.java similarity index 67% rename from inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/SpanSuggestionSupport.java rename to inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionSupport.java index 9a4da06ee85..b97b01aac50 100644 --- a/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/service/SpanSuggestionSupport.java +++ b/inception/inception-recommendation/src/main/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionSupport.java @@ -15,16 +15,19 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package de.tudarmstadt.ukp.inception.recommendation.service; +package de.tudarmstadt.ukp.inception.recommendation.span; import static de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion.FLAG_OVERLAP; import static de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion.FLAG_SKIPPED; import static de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion.FLAG_TRANSIENT_REJECTED; import static de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction.REJECTED; import static de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction.SKIPPED; +import static java.lang.Math.max; +import static java.lang.Math.min; import static java.util.Comparator.comparingInt; import static java.util.stream.Collectors.toList; import static org.apache.uima.cas.text.AnnotationPredicates.colocated; +import static org.apache.uima.fit.util.CasUtil.getType; import static org.apache.uima.fit.util.CasUtil.select; import java.lang.invoke.MethodHandles; @@ -34,6 +37,7 @@ import java.util.List; import java.util.Map; import java.util.Objects; +import java.util.Optional; import java.util.TreeMap; import org.apache.commons.collections4.multimap.ArrayListValuedHashMap; @@ -50,12 +54,18 @@ import org.springframework.context.ApplicationEventPublisher; import org.springframework.lang.Nullable; +import de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode; import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; +import de.tudarmstadt.ukp.dkpro.core.api.segmentation.TrimUtils; +import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Sentence; +import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Token; import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanAdapter; +import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanLayerSupport; import de.tudarmstadt.ukp.inception.recommendation.api.LearningRecordService; import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; +import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionRenderer; import de.tudarmstadt.ukp.inception.recommendation.api.SuggestionSupport_ImplBase; import de.tudarmstadt.ukp.inception.recommendation.api.event.RecommendationRejectedEvent; import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; @@ -63,33 +73,53 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordChangeLocation; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction; import de.tudarmstadt.ukp.inception.recommendation.api.model.Offset; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionGroup; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.ExtractionContext; +import de.tudarmstadt.ukp.inception.recommendation.config.RecommenderProperties; import de.tudarmstadt.ukp.inception.recommendation.util.OverlapIterator; import de.tudarmstadt.ukp.inception.schema.api.AnnotationSchemaService; import de.tudarmstadt.ukp.inception.schema.api.adapter.AnnotationException; import de.tudarmstadt.ukp.inception.schema.api.adapter.TypeAdapter; +import de.tudarmstadt.ukp.inception.schema.api.feature.FeatureSupportRegistry; public class SpanSuggestionSupport - extends SuggestionSupport_ImplBase + extends SuggestionSupport_ImplBase { private static final Logger LOG = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass()); public static final String TYPE = "SPAN"; + private final FeatureSupportRegistry featureSupportRegistry; + private final RecommenderProperties recommenderProperties; + public SpanSuggestionSupport(RecommendationService aRecommendationService, LearningRecordService aLearningRecordService, ApplicationEventPublisher aApplicationEventPublisher, - AnnotationSchemaService aSchemaService) + AnnotationSchemaService aSchemaService, FeatureSupportRegistry aFeatureSupportRegistry, + RecommenderProperties aRecommenderProperties) { super(aRecommendationService, aLearningRecordService, aApplicationEventPublisher, aSchemaService); + featureSupportRegistry = aFeatureSupportRegistry; + recommenderProperties = aRecommenderProperties; } @Override - public boolean accepts(AnnotationSuggestion aContext) + public boolean accepts(Recommender aContext) { - return aContext instanceof SpanSuggestion; + if (!SpanLayerSupport.TYPE.equals(aContext.getLayer().getType())) { + return false; + } + + var feature = aContext.getFeature(); + if (CAS.TYPE_NAME_STRING.equals(feature.getType()) || feature.isVirtualFeature()) { + return true; + } + + return false; } @Override @@ -432,4 +462,195 @@ var record = new LearningRecord(); record.setAnnotationFeature(aFeature); return record; } + + @Override + public Optional getRenderer() + { + return Optional.of(new SpanSuggestionRenderer(recommendationService, schemaService, + featureSupportRegistry, recommenderProperties)); + } + + @Override + public List extractSuggestions(ExtractionContext ctx) + { + var result = new ArrayList(); + for (var predictedFS : ctx.getPredictionCas(). select(ctx.getPredictedType())) { + if (!predictedFS.getBooleanValue(ctx.getPredictionFeature())) { + continue; + } + + var anchoringMode = ctx.getLayer().getAnchoringMode(); + var targetOffsets = getOffsets(anchoringMode, ctx.getOriginalCas(), predictedFS); + if (targetOffsets.isEmpty()) { + LOG.debug("Prediction cannot be anchored to [{}]: {}", anchoringMode, predictedFS); + continue; + } + + var autoAcceptMode = getAutoAcceptMode(predictedFS, ctx.getModeFeature()); + var labels = getPredictedLabels(predictedFS, ctx.getLabelFeature(), + ctx.isMultiLabels()); + var score = predictedFS.getDoubleValue(ctx.getScoreFeature()); + var scoreExplanation = predictedFS.getStringValue(ctx.getScoreExplanationFeature()); + var offsets = targetOffsets.get(); + var coveredText = ctx.getDocumentText().substring(offsets.getBegin(), offsets.getEnd()); + + for (var label : labels) { + var suggestion = SpanSuggestion.builder() // + .withId(RelationSuggestion.NEW_ID) // + .withGeneration(ctx.getGeneration()) // + .withRecommender(ctx.getRecommender()) // + .withDocument(ctx.getDocument()) // + .withPosition(offsets) // + .withCoveredText(coveredText) // + .withLabel(label) // + .withUiLabel(label) // + .withScore(score) // + .withScoreExplanation(scoreExplanation) // + .withAutoAcceptMode(autoAcceptMode) // + .build(); + result.add(suggestion); + } + } + return result; + } + + /** + * Calculates the offsets of the given predicted annotation in the original CAS . + * + * @param aMode + * the anchoring mode of the target layer + * @param aOriginalCas + * the original CAS. + * @param aPredictedAnnotation + * the predicted annotation. + * @return the proper offsets. + */ + static Optional getOffsets(AnchoringMode aMode, CAS aOriginalCas, + Annotation aPredictedAnnotation) + { + switch (aMode) { + case CHARACTERS: { + return getOffsetsAnchoredOnCharacters(aOriginalCas, aPredictedAnnotation); + } + case SINGLE_TOKEN: { + return getOffsetsAnchoredOnSingleTokens(aOriginalCas, aPredictedAnnotation); + } + case TOKENS: { + return getOffsetsAnchoredOnTokens(aOriginalCas, aPredictedAnnotation); + } + case SENTENCES: { + return getOffsetsAnchoredOnSentences(aOriginalCas, aPredictedAnnotation); + } + default: + throw new IllegalStateException("Unsupported anchoring mode: [" + aMode + "]"); + } + } + + private static Optional getOffsetsAnchoredOnCharacters(CAS aOriginalCas, + Annotation aPredictedAnnotation) + { + int[] offsets = { max(aPredictedAnnotation.getBegin(), 0), + min(aOriginalCas.getDocumentText().length(), aPredictedAnnotation.getEnd()) }; + TrimUtils.trim(aPredictedAnnotation.getCAS().getDocumentText(), offsets); + var begin = offsets[0]; + var end = offsets[1]; + return Optional.of(new Offset(begin, end)); + } + + private static Optional getOffsetsAnchoredOnSentences(CAS aOriginalCas, + Annotation aPredictedAnnotation) + { + var sentences = aOriginalCas.select(Sentence.class) // + .coveredBy(aPredictedAnnotation) // + .asList(); + + if (sentences.isEmpty()) { + // This can happen if a recommender uses different token boundaries (e.g. if a + // remote service performs its own tokenization). We might be smart here by + // looking for overlapping sentences instead of covered sentences. + LOG.trace("Discarding suggestion because no covered sentences were found: {}", + aPredictedAnnotation); + return Optional.empty(); + } + + var begin = sentences.get(0).getBegin(); + var end = sentences.get(sentences.size() - 1).getEnd(); + return Optional.of(new Offset(begin, end)); + } + + private static Optional getOffsetsAnchoredOnSingleTokens(CAS aOriginalCas, + Annotation aPredictedAnnotation) + { + Type tokenType = getType(aOriginalCas, Token.class); + var tokens = aOriginalCas. select(tokenType) // + .coveredBy(aPredictedAnnotation) // + .limit(2).asList(); + + if (tokens.isEmpty()) { + // This can happen if a recommender uses different token boundaries (e.g. if a + // remote service performs its own tokenization). We might be smart here by + // looking for overlapping tokens instead of contained tokens. + LOG.trace("Discarding suggestion because no covering token was found: {}", + aPredictedAnnotation); + return Optional.empty(); + } + + if (tokens.size() > 1) { + // We only want to accept single-token suggestions + LOG.trace("Discarding suggestion because only single-token suggestions are " + + "accepted: {}", aPredictedAnnotation); + return Optional.empty(); + } + + AnnotationFS token = tokens.get(0); + var begin = token.getBegin(); + var end = token.getEnd(); + return Optional.of(new Offset(begin, end)); + } + + public static Optional getOffsetsAnchoredOnTokens(CAS aOriginalCas, + Annotation aPredictedAnnotation) + { + var tokens = aOriginalCas.select(Token.class) // + .coveredBy(aPredictedAnnotation) // + .asList(); + + if (tokens.isEmpty()) { + if (aPredictedAnnotation.getBegin() == aPredictedAnnotation.getEnd()) { + var pos = aPredictedAnnotation.getBegin(); + var allTokens = aOriginalCas.select(Token.class).asList(); + Token prevToken = null; + for (var token : allTokens) { + if (prevToken == null && pos < token.getBegin()) { + return Optional.of(new Offset(token.getBegin(), token.getBegin())); + } + + if (token.covering(aPredictedAnnotation)) { + return Optional.of(new Offset(pos, pos)); + } + + if (prevToken != null && pos < token.getBegin()) { + return Optional.of(new Offset(prevToken.getEnd(), prevToken.getEnd())); + } + + prevToken = token; + } + + if (prevToken != null && pos >= prevToken.getEnd()) { + return Optional.of(new Offset(prevToken.getEnd(), prevToken.getEnd())); + } + } + + // This can happen if a recommender uses different token boundaries (e.g. if a + // remote service performs its own tokenization). We might be smart here by + // looking for overlapping tokens instead of covered tokens. + LOG.trace("Discarding suggestion because no covered tokens were found: {}", + aPredictedAnnotation); + return Optional.empty(); + } + + var begin = tokens.get(0).getBegin(); + var end = tokens.get(tokens.size() - 1).getEnd(); + return Optional.of(new Offset(begin, end)); + } } diff --git a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RelationSuggestionVisibilityCalculationTest.java b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/relation/RelationSuggestionVisibilityCalculationTest.java similarity index 98% rename from inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RelationSuggestionVisibilityCalculationTest.java rename to inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/relation/RelationSuggestionVisibilityCalculationTest.java index 4e104edcf98..7738e321921 100644 --- a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RelationSuggestionVisibilityCalculationTest.java +++ b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/relation/RelationSuggestionVisibilityCalculationTest.java @@ -15,7 +15,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package de.tudarmstadt.ukp.inception.recommendation.service; +package de.tudarmstadt.ukp.inception.recommendation.relation; import static de.tudarmstadt.ukp.inception.recommendation.service.Fixtures.getInvisibleSuggestions; import static de.tudarmstadt.ukp.inception.recommendation.service.Fixtures.getVisibleSuggestions; @@ -76,7 +76,7 @@ public void setUp() throws Exception when(annoService.listSupportedFeatures(layer)).thenReturn(asList(feature)); - sut = new RelationSuggestionSupport(null, learningRecordService, null, annoService); + sut = new RelationSuggestionSupport(null, learningRecordService, null, annoService, null); } @Test diff --git a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/Fixtures.java b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/Fixtures.java index 4e7d92aaf77..4bc0266f1a4 100644 --- a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/Fixtures.java +++ b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/Fixtures.java @@ -17,17 +17,15 @@ */ package de.tudarmstadt.ukp.inception.recommendation.service; -import static de.tudarmstadt.ukp.inception.recommendation.api.model.AutoAcceptMode.NEVER; -import static java.util.stream.Collectors.toList; - import java.util.ArrayList; import java.util.Collection; import java.util.List; import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; -import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationPosition; import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionDocumentGroup; @@ -38,72 +36,60 @@ public class Fixtures // AnnotationSuggestion private final static long RECOMMENDER_ID = 1; private final static String RECOMMENDER_NAME = "TestEntityRecommender"; - private final static String UI_LABEL = "TestUiLabel"; private final static double CONFIDENCE = 0.2; private final static String CONFIDENCE_EXPLANATION = "Predictor A: 0.05 | Predictor B: 0.15"; private final static String COVERED_TEXT = "TestText"; - static List getInvisibleSuggestions( + public static List getInvisibleSuggestions( Collection> aSuggestions) { return aSuggestions.stream() // .flatMap(SuggestionGroup::stream) // .filter(s -> !s.isVisible()) // - .collect(toList()); + .toList(); } - static List getVisibleSuggestions( + public static List getVisibleSuggestions( Collection> aSuggestions) { return aSuggestions.stream() // .flatMap(SuggestionGroup::stream) // .filter(s -> s.isVisible()) // - .collect(toList()); + .toList(); } - static SuggestionDocumentGroup makeSpanSuggestionGroup(SourceDocument doc, - AnnotationFeature aFeat, int[][] vals) + public static SuggestionDocumentGroup makeSpanSuggestionGroup( + SourceDocument doc, AnnotationFeature aFeat, int[][] vals) { + var rec = Recommender.builder().withId(RECOMMENDER_ID).withName(RECOMMENDER_NAME) + .withLayer(aFeat.getLayer()).withFeature(aFeat).build(); + List suggestions = new ArrayList<>(); for (int[] val : vals) { - suggestions.add(new SpanSuggestion(val[0], RECOMMENDER_ID, RECOMMENDER_NAME, - aFeat.getLayer().getId(), aFeat.getName(), doc.getName(), val[1], val[2], - COVERED_TEXT, null, UI_LABEL, CONFIDENCE, CONFIDENCE_EXPLANATION, NEVER)); + var suggestion = SpanSuggestion.builder().withId(val[0]).withRecommender(rec) + .withDocument(doc).withPosition(val[1], val[2]).withCoveredText(COVERED_TEXT) + .withScore(CONFIDENCE).withScoreExplanation(CONFIDENCE_EXPLANATION).build(); + suggestions.add(suggestion); } - return new SuggestionDocumentGroup<>(suggestions); + return SuggestionDocumentGroup.groupsOfType(SpanSuggestion.class, suggestions); } - static SuggestionDocumentGroup makeRelationSuggestionGroup( + public static SuggestionDocumentGroup makeRelationSuggestionGroup( SourceDocument doc, AnnotationFeature aFeat, int[][] vals) { + var rec = Recommender.builder().withId(RECOMMENDER_ID).withName(RECOMMENDER_NAME) + .withLayer(aFeat.getLayer()).withFeature(aFeat).build(); + List suggestions = new ArrayList<>(); for (int[] val : vals) { - suggestions.add(new RelationSuggestion(val[0], RECOMMENDER_ID, RECOMMENDER_NAME, - aFeat.getLayer().getId(), aFeat.getName(), doc.getName(), val[1], val[2], - val[3], val[4], null, UI_LABEL, CONFIDENCE, CONFIDENCE_EXPLANATION, NEVER)); + var suggestion = RelationSuggestion.builder().withId(val[0]).withRecommender(rec) + .withDocument(doc) + .withPosition(new RelationPosition(val[1], val[2], val[3], val[4])) + .withScore(CONFIDENCE).withScoreExplanation(CONFIDENCE_EXPLANATION).build(); + suggestions.add(suggestion); } - return new SuggestionDocumentGroup<>(suggestions); - } - - public static SpanSuggestion makeSuggestion(int aBegin, int aEnd, String aLabel, - SourceDocument aDoc, AnnotationLayer aLayer, AnnotationFeature aFeature) - { - return new SpanSuggestion(0, // aId, - 0, // aRecommenderId, - "", // aRecommenderName - aLayer.getId(), // aLayerId, - aFeature.getName(), // aFeature, - aDoc.getName(), // aDocumentName - aBegin, // aBegin - aEnd, // aEnd - "", // aCoveredText, - aLabel, // aLabel - aLabel, // aUiLabel - 0.0, // aScore - "", // aScoreExplanation, - NEVER // autoAccept - ); + return SuggestionDocumentGroup.groupsOfType(RelationSuggestion.class, suggestions); } } diff --git a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImplIntegrationTest.java b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImplIntegrationTest.java index 0e366ad98c2..8821b35fd9d 100644 --- a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImplIntegrationTest.java +++ b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImplIntegrationTest.java @@ -23,11 +23,9 @@ import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_IS_PREDICTION; import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_SCORE_EXPLANATION_SUFFIX; import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_SCORE_SUFFIX; -import static de.tudarmstadt.ukp.inception.recommendation.api.model.AutoAcceptMode.NEVER; import static de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordChangeLocation.DETAIL_EDITOR; import static de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordChangeLocation.MAIN_EDITOR; import static de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction.ACCEPTED; -import static de.tudarmstadt.ukp.inception.recommendation.service.SuggestionExtraction.getOffsetsAnchoredOnTokens; import static java.util.Arrays.asList; import static org.apache.uima.fit.factory.JCasFactory.createJCas; import static org.apache.uima.fit.factory.JCasFactory.createText; @@ -43,10 +41,8 @@ import org.apache.uima.cas.CAS; import org.apache.uima.cas.Feature; import org.apache.uima.cas.Type; -import org.apache.uima.fit.testing.factory.TokenBuilder; import org.apache.uima.fit.util.CasUtil; import org.apache.uima.jcas.JCas; -import org.apache.uima.jcas.tcas.Annotation; import org.apache.uima.resource.metadata.TypeSystemDescription; import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.BeforeEach; @@ -67,9 +63,8 @@ import de.tudarmstadt.ukp.clarin.webanno.model.Project; import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; import de.tudarmstadt.ukp.dkpro.core.api.ner.type.NamedEntity; -import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Sentence; -import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Token; import de.tudarmstadt.ukp.inception.annotation.feature.string.StringFeatureSupport; +import de.tudarmstadt.ukp.inception.annotation.layer.relation.RelationLayerSupport; import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanAdapter; import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanLayerSupport; import de.tudarmstadt.ukp.inception.annotation.storage.CasStorageSession; @@ -78,9 +73,12 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction; import de.tudarmstadt.ukp.inception.recommendation.api.model.Offset; import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationPosition; import de.tudarmstadt.ukp.inception.recommendation.api.model.RelationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.recommender.RecommendationEngineFactory; +import de.tudarmstadt.ukp.inception.recommendation.relation.RelationSuggestionSupport; +import de.tudarmstadt.ukp.inception.recommendation.span.SpanSuggestionSupport; import de.tudarmstadt.ukp.inception.schema.api.layer.LayerSupportRegistry; import de.tudarmstadt.ukp.inception.schema.service.AnnotationSchemaServiceImpl; import de.tudarmstadt.ukp.inception.schema.service.FeatureSupportRegistryImpl; @@ -110,16 +108,16 @@ public class RecommendationServiceImplIntegrationTest private FeatureSupportRegistryImpl featureSupportRegistry; private LayerRecommendtionSupportRegistryImpl layerRecommendtionSupportRegistry; private Project project; - private AnnotationLayer layer; - private Recommender rec; - private AnnotationFeature feature; + private AnnotationLayer spanLayer; + private Recommender spanLayerRecommender; + private AnnotationFeature spanLayerFeature; @BeforeEach public void setUp() throws Exception { layerRecommendtionSupportRegistry = new LayerRecommendtionSupportRegistryImpl(asList( // - new SpanSuggestionSupport(sut, sut, null, schemaService), - new RelationSuggestionSupport(sut, sut, null, schemaService))); + new SpanSuggestionSupport(sut, sut, null, schemaService, null, null), + new RelationSuggestionSupport(sut, sut, null, schemaService, null))); layerRecommendtionSupportRegistry.init(); sut = new RecommendationServiceImpl(null, null, null, recommenderFactoryRegistry, null, @@ -130,12 +128,11 @@ public void setUp() throws Exception featureSupportRegistry.init(); project = createProject(PROJECT_NAME); - layer = createAnnotationLayer(); - layer.setProject(project); - feature = createAnnotationFeature(layer, "value"); + spanLayer = createSpanLayer(NamedEntity._TypeName); + spanLayerFeature = createAnnotationFeature(spanLayer, "value"); - rec = buildRecommender(project, feature); - sut.createOrUpdateRecommender(rec); + spanLayerRecommender = buildRecommender(spanLayerFeature); + sut.createOrUpdateRecommender(spanLayerRecommender); } @AfterEach @@ -152,13 +149,13 @@ public void thatApplicationContextStarts() @Test public void listRecommenders_WithOneEnabledRecommender_ShouldReturnStoredRecommender() { - sut.createOrUpdateRecommender(rec); + sut.createOrUpdateRecommender(spanLayerRecommender); - var enabledRecommenders = sut.listEnabledRecommenders(rec.getLayer()); + var enabledRecommenders = sut.listEnabledRecommenders(spanLayerRecommender.getLayer()); assertThat(enabledRecommenders) // .as("Check that the previously created recommender is found") // - .containsExactly(rec); + .containsExactly(spanLayerRecommender); } @SuppressWarnings("unchecked") @@ -171,7 +168,7 @@ public void getNumOfEnabledRecommenders_WithOneEnabledRecommender() assertThat(recommenderFactoryRegistry.getFactory("nummy")).isNotNull(); - sut.createOrUpdateRecommender(rec); + sut.createOrUpdateRecommender(spanLayerRecommender); assertThat(sut.countEnabledRecommenders()).isEqualTo(1); } @@ -179,8 +176,8 @@ public void getNumOfEnabledRecommenders_WithOneEnabledRecommender() @Test public void getNumOfEnabledRecommenders_WithNoEnabledRecommender() { - rec.setEnabled(false); - testEntityManager.persist(rec); + spanLayerRecommender.setEnabled(false); + testEntityManager.persist(spanLayerRecommender); assertThat(sut.countEnabledRecommenders()).isEqualTo(0); } @@ -188,18 +185,18 @@ public void getNumOfEnabledRecommenders_WithNoEnabledRecommender() @Test public void getRecommenders_WithOneEnabledRecommender_ShouldReturnStoredRecommender() { - assertThat(sut.getEnabledRecommender(rec.getId())) + assertThat(sut.getEnabledRecommender(spanLayerRecommender.getId())) .as("Check that only the previously created recommender is found").isPresent() - .contains(rec); + .contains(spanLayerRecommender); } @Test public void getRecommenders_WithOnlyDisabledRecommender_ShouldReturnEmptyList() { - rec.setEnabled(false); - testEntityManager.persist(rec); + spanLayerRecommender.setEnabled(false); + testEntityManager.persist(spanLayerRecommender); - assertThat(sut.getEnabledRecommender(rec.getId())) // + assertThat(sut.getEnabledRecommender(spanLayerRecommender.getId())) // .as("Check that no recommender is found") // .isEmpty(); } @@ -222,13 +219,13 @@ public void monkeyPatchTypeSystem_WithNer_CreatesScoreFeatures() throws Exceptio when(schemaService.getFullProjectTypeSystem(project)) .thenReturn(typeSystem2TypeSystemDescription(jCas.getTypeSystem())); - when(schemaService.listAnnotationFeature(project)).thenReturn(asList(feature)); + when(schemaService.listAnnotationFeature(project)).thenReturn(asList(spanLayerFeature)); doCallRealMethod().when(schemaService).upgradeCas(any(CAS.class), any(CAS.class), any(TypeSystemDescription.class)); sut.cloneAndMonkeyPatchCAS(project, jCas.getCas(), jCas.getCas()); - Type type = CasUtil.getType(jCas.getCas(), layer.getName()); + Type type = CasUtil.getType(jCas.getCas(), spanLayer.getName()); assertThat(type.getFeatures()) // .extracting(Feature::getShortName) // @@ -237,86 +234,53 @@ public void monkeyPatchTypeSystem_WithNer_CreatesScoreFeatures() throws Exceptio "begin", // "end", // "value", // - feature.getName() + FEATURE_NAME_SCORE_SUFFIX, // - feature.getName() + FEATURE_NAME_SCORE_EXPLANATION_SUFFIX, // - feature.getName() + FEATURE_NAME_AUTO_ACCEPT_MODE_SUFFIX, // + spanLayerFeature.getName() + FEATURE_NAME_SCORE_SUFFIX, // + spanLayerFeature.getName() + FEATURE_NAME_SCORE_EXPLANATION_SUFFIX, // + spanLayerFeature.getName() + FEATURE_NAME_AUTO_ACCEPT_MODE_SUFFIX, // "identifier", // FEATURE_NAME_IS_PREDICTION); } } - @Test - void thatZeroWithAnnotationsAreCorrectlyAnchoredOnTokens() throws Exception - { - var jCas = createJCas(); - TokenBuilder.create(Token.class, Sentence.class).buildTokens(jCas, " This is a test. "); - var textLength = jCas.getDocumentText().length(); - var tokens = jCas.select(Token.class).asList(); - var firstTokenBegin = tokens.get(0).getBegin(); - var lastTokenEnd = tokens.get(tokens.size() - 1).getEnd(); - - assertThat(getOffsetsAnchoredOnTokens(jCas.getCas(), new Annotation(jCas, 0, 0))).get() - .as("Zero-width annotation before first token snaps to first token start") - .isEqualTo(new Offset(firstTokenBegin, firstTokenBegin)); - - assertThat(getOffsetsAnchoredOnTokens(jCas.getCas(), - new Annotation(jCas, textLength, textLength))).get() - .as("Zero-width annotation after last token snaps to last token end") - .isEqualTo(new Offset(lastTokenEnd, lastTokenEnd)); - - assertThat(getOffsetsAnchoredOnTokens(jCas.getCas(), new Annotation(jCas, 4, 4))).get() - .as("Zero-width annotation within token remains") // - .isEqualTo(new Offset(4, 4)); - - assertThat(getOffsetsAnchoredOnTokens(jCas.getCas(), new Annotation(jCas, 10, 10))).get() - .as("Zero-width annotation between tokens snaps to end of previous") // - .isEqualTo(new Offset(9, 9)); - } - @Test void testUpsertSpanFeature() throws Exception { var docOwner = "dummy"; var doc = SourceDocument.builder() // + .withId(1l) // .withProject(project) // .build(); - var feature = AnnotationFeature.builder() // - .withName(NamedEntity._FeatName_value) // - .withType(CAS.TYPE_NAME_STRING) // - .build(); - var layer = AnnotationLayer.builder() // - .forJCasClass(NamedEntity.class) // - .build(); - var adapter = new SpanAdapter(layerSupportRegistry, featureSupportRegistry, null, layer, + var adapter = new SpanAdapter(layerSupportRegistry, featureSupportRegistry, null, spanLayer, () -> asList(), asList()); - when(schemaService.getLayer(anyLong())).thenReturn(layer); + when(schemaService.getLayer(anyLong())).thenReturn(spanLayer); when(schemaService.getAdapter(any())).thenReturn(adapter); - when(schemaService.getFeature(any(), any())).thenReturn(feature); + when(schemaService.getFeature(any(), any())).thenReturn(spanLayerFeature); - layer.setOverlapMode(NO_OVERLAP); + spanLayer.setOverlapMode(NO_OVERLAP); var cas = createJCas(); var targetFS = new NamedEntity(cas, 0, 10); targetFS.addToIndexes(); assertThat(targetFS.getValue()).isNull(); - var s1 = SpanSuggestion.builder().withLabel("V1").withPosition(new Offset(targetFS)) - .build(); + var s1 = SpanSuggestion.builder().withLabel("V1").withRecommender(spanLayerRecommender) + .withPosition(new Offset(targetFS)).build(); sut.acceptSuggestion(USER_NAME, doc, docOwner, cas.getCas(), s1, MAIN_EDITOR); assertThat(targetFS.getValue()) // .as("Label was merged into existing annotation replacing unset label") // .isEqualTo("V1"); - var s2 = SpanSuggestion.builder().withLabel("V2").withPosition(new Offset(targetFS)) - .build(); + var s2 = SpanSuggestion.builder().withLabel("V2").withRecommender(spanLayerRecommender) + .withPosition(new Offset(targetFS)).build(); sut.acceptSuggestion(USER_NAME, doc, docOwner, cas.getCas(), s2, MAIN_EDITOR); assertThat(targetFS.getValue()) // .as("Label was merged into existing annotation replacing previous label") // .isEqualTo("V2"); - var s3 = SpanSuggestion.builder().withLabel("V3").withPosition(new Offset(10, 20)).build(); + var s3 = SpanSuggestion.builder().withLabel("V3").withRecommender(spanLayerRecommender) + .withPosition(new Offset(10, 20)).build(); sut.acceptSuggestion(USER_NAME, doc, docOwner, cas.getCas(), s3, MAIN_EDITOR); assertThat(cas.select(NamedEntity.class).asList()) // @@ -326,22 +290,22 @@ void testUpsertSpanFeature() throws Exception tuple(0, 10, "V2"), // tuple(10, 20, "V3")); - layer.setOverlapMode(ANY_OVERLAP); + spanLayer.setOverlapMode(ANY_OVERLAP); cas.reset(); targetFS = new NamedEntity(cas, 0, 10); targetFS.addToIndexes(); assertThat(targetFS.getValue()).isNull(); - var s4 = SpanSuggestion.builder().withLabel("V1").withPosition(new Offset(targetFS)) - .build(); + var s4 = SpanSuggestion.builder().withLabel("V1").withRecommender(spanLayerRecommender) + .withPosition(new Offset(targetFS)).build(); sut.acceptSuggestion(USER_NAME, doc, docOwner, cas.getCas(), s4, MAIN_EDITOR); assertThat(targetFS.getValue()) // .as("Label was merged into existing annotation replacing unset label") // .isEqualTo("V1"); - var s5 = SpanSuggestion.builder().withLabel("V2").withPosition(new Offset(targetFS)) - .build(); + var s5 = SpanSuggestion.builder().withLabel("V2").withRecommender(spanLayerRecommender) + .withPosition(new Offset(targetFS)).build(); sut.acceptSuggestion(USER_NAME, doc, docOwner, cas.getCas(), s5, MAIN_EDITOR); assertThat(cas.select(NamedEntity.class).asList()) // @@ -351,7 +315,8 @@ void testUpsertSpanFeature() throws Exception tuple(0, 10, "V1"), // tuple(0, 10, "V2")); - var s6 = SpanSuggestion.builder().withLabel("V3").withPosition(new Offset(10, 20)).build(); + var s6 = SpanSuggestion.builder().withLabel("V3").withRecommender(spanLayerRecommender) + .withPosition(new Offset(10, 20)).build(); sut.acceptSuggestion(USER_NAME, doc, docOwner, cas.getCas(), s6, MAIN_EDITOR); assertThat(cas.select(NamedEntity.class).asList()) // @@ -365,7 +330,8 @@ void testUpsertSpanFeature() throws Exception new NamedEntity(cas, 0, 10).addToIndexes(); new NamedEntity(cas, 0, 10).addToIndexes(); - var s7 = SpanSuggestion.builder().withLabel("V4").withPosition(new Offset(0, 10)).build(); + var s7 = SpanSuggestion.builder().withLabel("V4").withRecommender(spanLayerRecommender) + .withPosition(new Offset(0, 10)).build(); sut.acceptSuggestion(USER_NAME, doc, docOwner, cas.getCas(), s7, MAIN_EDITOR); assertThat(cas.select(NamedEntity.class).asList()) // @@ -383,36 +349,31 @@ void testUpsertSpanFeature() throws Exception public void thatSpanSuggestionsCanBeRecorded() { var sourceDoc = createSourceDocument("doc"); - var layer = createAnnotationLayer("layer"); - var feature = createAnnotationFeature(layer, FEATURE_NAME); var suggestion = SpanSuggestion.builder() // .withId(42) // - .withRecommenderId(1337) // - .withRecommenderName("testRecommender") // - .withLayerId(layer.getId()) // - .withFeature(feature.getName()) // - .withDocumentName(sourceDoc.getName()) // - .withPosition(new Offset(7, 14)) // + .withRecommender(spanLayerRecommender) // + .withDocument(sourceDoc) // + .withPosition(7, 14) // .withCoveredText("aCoveredText") // .withLabel("testLabel") // .withUiLabel("testUiLabel") // .withScore(0.42) // .withScoreExplanation("Test confidence") // - .withAutoAcceptMode(NEVER) // .build(); - sut.logRecord(USER_NAME, sourceDoc, USER_NAME, suggestion, feature, ACCEPTED, MAIN_EDITOR); + sut.logRecord(USER_NAME, sourceDoc, USER_NAME, suggestion, spanLayerFeature, ACCEPTED, + MAIN_EDITOR); - var records = sut.listLearningRecords(USER_NAME, USER_NAME, layer); + var records = sut.listLearningRecords(USER_NAME, USER_NAME, spanLayer); assertThat(records).hasSize(1); LearningRecord record = records.get(0); assertThat(record).hasFieldOrProperty("id") // .hasFieldOrPropertyWithValue("sourceDocument", sourceDoc) // .hasFieldOrPropertyWithValue("user", USER_NAME) // - .hasFieldOrPropertyWithValue("layer", layer) // - .hasFieldOrPropertyWithValue("annotationFeature", feature) // + .hasFieldOrPropertyWithValue("layer", spanLayer) // + .hasFieldOrPropertyWithValue("annotationFeature", spanLayerFeature) // .hasFieldOrPropertyWithValue("offsetBegin", 7) // .hasFieldOrPropertyWithValue("offsetEnd", 14) // .hasFieldOrPropertyWithValue("offsetBegin2", -1) // @@ -428,12 +389,14 @@ public void thatSpanSuggestionsCanBeRecorded() public void thatRelationSuggestionsCanBeRecorded() { var sourceDoc = createSourceDocument("doc"); - var layer = createAnnotationLayer("layer"); + var layer = createRelationLayer("layer"); var feature = createAnnotationFeature(layer, FEATURE_NAME); + var rec = buildRecommender(feature); - var suggestion = new RelationSuggestion(42, 1337, "testRecommender", layer.getId(), - feature.getName(), sourceDoc.getName(), 7, 14, 21, 28, "testLabel", "testUiLabel", - 0.42, "Test confidence", NEVER); + var suggestion = RelationSuggestion.builder().withId(42).withRecommender(rec) + .withDocument(sourceDoc).withPosition(new RelationPosition(7, 14, 21, 28)) + .withLabel("testLabel").withUiLabel("testUiLabel").withScore(0.42) + .withScoreExplanation("Test confidence").build(); sut.logRecord(USER_NAME, sourceDoc, USER_NAME, suggestion, feature, LearningRecordUserAction.REJECTED, DETAIL_EDITOR); @@ -463,33 +426,36 @@ void thatListingRecordsForRendering() { var sourceDoc1 = createSourceDocument("doc1"); var sourceDoc2 = createSourceDocument("doc2"); - var layer1 = createAnnotationLayer("layer1"); - var layer2 = createAnnotationLayer("layer2"); + var layer1 = createSpanLayer("layer1"); + var layer2 = createSpanLayer("layer2"); var feature1 = createAnnotationFeature(layer1, "feat1"); var feature2 = createAnnotationFeature(layer2, "feat1"); + var rec1 = buildRecommender(feature1); + var rec2 = buildRecommender(feature2); + Offset position = new Offset(7, 14); sut.logRecord(USER_NAME, sourceDoc1, USER_NAME, - new SpanSuggestion(42, 1337, "testRecommender", layer1.getId(), feature1.getName(), - sourceDoc1.getName(), 7, 14, "aCoveredText", "testLabel", "testUiLabel", - 0.42, "Test confidence", NEVER), + SpanSuggestion.builder().withRecommender(rec1).withDocument(sourceDoc1) + .withPosition(position).withLabel("testLabel") + .withCoveredText("aCoveredText").build(), feature1, ACCEPTED, MAIN_EDITOR); sut.logRecord(USER_NAME, sourceDoc1, USER_NAME, - new SpanSuggestion(42, 1337, "testRecommender2", layer2.getId(), feature2.getName(), - sourceDoc1.getName(), 7, 14, "aCoveredText", "testLabel", "testUiLabel", - 0.42, "Test confidence", NEVER), + SpanSuggestion.builder().withRecommender(rec2).withDocument(sourceDoc1) + .withPosition(position).withLabel("testLabel") + .withCoveredText("aCoveredText").build(), feature2, ACCEPTED, MAIN_EDITOR); sut.logRecord(USER_NAME, sourceDoc2, USER_NAME, - new SpanSuggestion(42, 1337, "testRecommender", layer1.getId(), feature1.getName(), - sourceDoc2.getName(), 7, 14, "aCoveredText", "testLabel", "testUiLabel", - 0.42, "Test confidence", NEVER), + SpanSuggestion.builder().withRecommender(rec1).withDocument(sourceDoc1) + .withPosition(position).withLabel("testLabel") + .withCoveredText("aCoveredText").build(), feature1, ACCEPTED, MAIN_EDITOR); sut.logRecord(USER_NAME, sourceDoc2, USER_NAME, - new SpanSuggestion(42, 1337, "testRecommender2", layer2.getId(), feature2.getName(), - sourceDoc2.getName(), 7, 14, "aCoveredText", "testLabel", "testUiLabel", - 0.42, "Test confidence", NEVER), + SpanSuggestion.builder().withRecommender(rec2).withDocument(sourceDoc1) + .withPosition(position).withLabel("testLabel") + .withCoveredText("aCoveredText").build(), feature2, ACCEPTED, MAIN_EDITOR); assertThat(sut.listLearningRecords(USER_NAME, sourceDoc1, USER_NAME, feature1)).hasSize(1); @@ -504,13 +470,11 @@ void thatListingRecordsForRendering() private SourceDocument createSourceDocument(String aName) { - var doc = new SourceDocument(); - doc.setProject(project); - doc.setName(aName); + var doc = SourceDocument.builder().withProject(project).withName(aName).build(); return testEntityManager.persist(doc); } - private AnnotationLayer createAnnotationLayer(String aType) + private AnnotationLayer createSpanLayer(String aType) { var l = new AnnotationLayer(); l.setProject(project); @@ -524,6 +488,20 @@ private AnnotationLayer createAnnotationLayer(String aType) return testEntityManager.persist(l); } + private AnnotationLayer createRelationLayer(String aType) + { + var l = new AnnotationLayer(); + l.setProject(project); + l.setEnabled(true); + l.setName(aType); + l.setReadonly(false); + l.setType(RelationLayerSupport.TYPE); + l.setUiName(aType); + l.setAnchoringMode(false, false); + + return testEntityManager.persist(l); + } + private AnnotationFeature createAnnotationFeature(AnnotationLayer aLayer, String aName) { var f = new AnnotationFeature(); @@ -543,22 +521,17 @@ private Project createProject(String aName) return testEntityManager.persist(l); } - private AnnotationLayer createAnnotationLayer() - { - return createAnnotationLayer(NamedEntity._TypeName); - } - - private Recommender buildRecommender(Project aProject, AnnotationFeature aFeature) + private Recommender buildRecommender(AnnotationFeature aFeature) { var r = new Recommender(); r.setLayer(aFeature.getLayer()); r.setFeature(aFeature); - r.setProject(aProject); + r.setProject(aFeature.getProject()); r.setAlwaysSelected(true); r.setSkipEvaluation(false); r.setMaxRecommendations(3); r.setTool("dummyRecommenderTool"); - return r; + return testEntityManager.persist(r); } } diff --git a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImplTest.java b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImplTest.java index f5b1e964ffc..140dc2bf272 100644 --- a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImplTest.java +++ b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/RecommendationServiceImplTest.java @@ -17,35 +17,12 @@ */ package de.tudarmstadt.ukp.inception.recommendation.service; -import static de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode.CHARACTERS; -import static de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode.SENTENCES; -import static de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode.SINGLE_TOKEN; -import static de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode.TOKENS; -import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_AUTO_ACCEPT_MODE_SUFFIX; -import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_IS_PREDICTION; -import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_SCORE_EXPLANATION_SUFFIX; -import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_SCORE_SUFFIX; -import static de.tudarmstadt.ukp.inception.recommendation.service.SuggestionExtraction.extractSuggestions; -import static de.tudarmstadt.ukp.inception.recommendation.service.SuggestionExtraction.getOffsets; -import static de.tudarmstadt.ukp.inception.support.uima.FeatureStructureBuilder.buildFS; -import static java.util.Arrays.asList; -import static org.apache.uima.cas.CAS.TYPE_NAME_ANNOTATION; -import static org.apache.uima.cas.CAS.TYPE_NAME_BOOLEAN; -import static org.apache.uima.cas.CAS.TYPE_NAME_DOUBLE; -import static org.apache.uima.cas.CAS.TYPE_NAME_STRING; -import static org.apache.uima.fit.factory.CasFactory.createCas; -import static org.apache.uima.fit.factory.TypeSystemDescriptionFactory.createTypeSystemDescription; -import static org.apache.uima.util.CasCreationUtils.mergeTypeSystems; import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.tuple; import java.util.Arrays; -import org.apache.uima.UIMAFramework; -import org.apache.uima.fit.factory.CasFactory; -import org.apache.uima.fit.factory.JCasFactory; import org.apache.uima.fit.testing.factory.TokenBuilder; -import org.apache.uima.jcas.tcas.Annotation; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; @@ -56,7 +33,6 @@ import de.tudarmstadt.ukp.clarin.webanno.security.model.User; import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Sentence; import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Token; -import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanLayerSupport; import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.Offset; import de.tudarmstadt.ukp.inception.recommendation.api.model.Predictions; @@ -70,9 +46,6 @@ class RecommendationServiceImplTest private SourceDocument doc1; private SourceDocument doc2; private AnnotationLayer layer1; - private AnnotationLayer layer2; - private AnnotationFeature feature1; - private AnnotationFeature feature2; @BeforeEach void setup() @@ -83,9 +56,6 @@ void setup() doc2 = new SourceDocument("doc2", null, null); doc2.setId(2l); layer1 = AnnotationLayer.builder().withId(1l).withName("layer1").build(); - layer2 = AnnotationLayer.builder().withId(2l).withName("layer2").build(); - feature1 = AnnotationFeature.builder().withName("feat1").build(); - feature2 = AnnotationFeature.builder().withName("feat2").build(); } @Test @@ -102,15 +72,15 @@ void testReconciliation() throws Exception var existingSuggestions = Arrays. asList( // SpanSuggestion.builder() // .withId(0) // - .withPosition(new Offset(0, 10)) // - .withDocumentName(doc.getName()) // + .withPosition(0, 10) // + .withDocument(doc) // .withLabel("aged") // .withRecommender(rec) // .build(), SpanSuggestion.builder() // .withId(1) // - .withPosition(new Offset(0, 10)) // - .withDocumentName(doc.getName()) // + .withPosition(0, 10) // + .withDocument(doc) // .withLabel("removed") // .withRecommender(rec) // .build()); @@ -120,15 +90,15 @@ void testReconciliation() throws Exception var newSuggestions = Arrays. asList( // SpanSuggestion.builder() // .withId(2) // - .withPosition(new Offset(0, 10)) // - .withDocumentName(doc.getName()) // + .withPosition(0, 10) // + .withDocument(doc) // .withLabel("aged") // .withRecommender(rec) // .build(), SpanSuggestion.builder() // .withId(3) // .withPosition(new Offset(0, 10)) // - .withDocumentName(doc.getName()) // + .withDocument(doc) // .withLabel("added") // .withRecommender(rec) // .build()); @@ -141,170 +111,4 @@ void testReconciliation() throws Exception AnnotationSuggestion::getAge) // .containsExactlyInAnyOrder(tuple(0, "aged", 1), tuple(3, "added", 0)); } - - @Test - void testOffsetAlignmentWithAnchoringOnCharacters() throws Exception - { - var mode = CHARACTERS; - var targetCas = CasFactory.createCas(); - tokenBuilder.buildTokens(targetCas.getJCas(), "This is a test ."); - - var suggestionCas = JCasFactory.createText(targetCas.getDocumentText()); - - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 1))) // - .as("Trivial case: one character") // - .get().isEqualTo(new Offset(0, 1)); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 4, 8))) // - .as("Leading and trailing space should be removed") // - .get().isEqualTo(new Offset(5, 7)); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, -10, 100))) // - .as("Range should be clipped to document boundaries").get() - .isEqualTo(new Offset(0, targetCas.getDocumentText().length())); - } - - @Test - void testOffsetAlignmentWithAnchoringOnSingleToken() throws Exception - { - var mode = SINGLE_TOKEN; - var targetCas = CasFactory.createCas(); - tokenBuilder.buildTokens(targetCas.getJCas(), "This is a test ."); - - var suggestionCas = JCasFactory.createText(targetCas.getDocumentText()); - - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 1))) // - .as("Reduce to empty if no token is fully covered") // - .isEmpty(); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 4))) // - .as("Trivial case: one token") // - .get().isEqualTo(new Offset(0, 4)); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 7))) // - .as("Discard suggestion if it covers more than one token") // - .isEmpty(); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, -10, 100))) // - .as("Discard suggestion if it covers more than one token (2)") // - .isEmpty(); - } - - @Test - void testOffsetAlignmentWithAnchoringOnTokens() throws Exception - { - var mode = TOKENS; - var targetCas = CasFactory.createCas(); - tokenBuilder.buildTokens(targetCas.getJCas(), "This is a test ."); - - var suggestionCas = JCasFactory.createText(targetCas.getDocumentText()); - - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 1))) // - .as("Reduce to empty if no token is fully covered") // - .isEmpty(); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 4))) // - .as("Trivial case: one token") // - .get().isEqualTo(new Offset(0, 4)); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 7))) // - .as("Trivial case: two tokens") // - .get().isEqualTo(new Offset(0, 7)); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 2, 12))) // - .as("Discard incompletely covered tokens") // - .get().isEqualTo(new Offset(5, 9)); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, -10, 100))) // - .as("Range should be clipped to document boundaries").get() - .isEqualTo(new Offset(0, targetCas.getDocumentText().length())); - } - - @Test - void testOffsetAlignmentWithAnchoringOnSentences() throws Exception - { - var mode = SENTENCES; - var targetCas = CasFactory.createCas(); - tokenBuilder.buildTokens(targetCas.getJCas(), "This is a test .\nAnother sentence here ."); - - var suggestionCas = JCasFactory.createText(targetCas.getDocumentText()); - var sentences = targetCas.select(Sentence.class).asList(); - var sent1 = sentences.get(0); - var sent2 = sentences.get(1); - - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 1))) // - .as("Reduce to empty if no sentence is fully covered") // - .isEmpty(); - assertThat(getOffsets(mode, targetCas, - new Annotation(suggestionCas, sent1.getBegin(), sent1.getEnd()))) // - .as("Trivial case: one sentence") // - .get().isEqualTo(new Offset(sent1.getBegin(), sent1.getEnd())); - assertThat(getOffsets(mode, targetCas, - new Annotation(suggestionCas, sent1.getBegin(), sent2.getEnd()))) // - .as("Trivial case: two sentences") // - .get().isEqualTo(new Offset(sent1.getBegin(), sent2.getEnd())); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 30))) // - .as("Discard incompletely covered sentences") // - .get().isEqualTo(new Offset(0, 16)); - assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, -10, 100))) // - .as("Range should be clipped to document boundaries").get() - .isEqualTo(new Offset(0, targetCas.getDocumentText().length())); - } - - @Test - void testExtractSuggestionsWithSpanSuggestions() throws Exception - { - var tsd = UIMAFramework.getResourceSpecifierFactory().createTypeSystemDescription(); - var predType = tsd.addType("Prediction", null, TYPE_NAME_ANNOTATION); - predType.addFeature("value", null, TYPE_NAME_STRING); - predType.addFeature("value" + FEATURE_NAME_SCORE_SUFFIX, null, TYPE_NAME_DOUBLE); - predType.addFeature("value" + FEATURE_NAME_SCORE_EXPLANATION_SUFFIX, null, - TYPE_NAME_STRING); - predType.addFeature("value" + FEATURE_NAME_AUTO_ACCEPT_MODE_SUFFIX, null, TYPE_NAME_STRING); - predType.addFeature(FEATURE_NAME_IS_PREDICTION, null, TYPE_NAME_BOOLEAN); - - var targetCas = createCas(mergeTypeSystems(asList(tsd, createTypeSystemDescription()))); - tokenBuilder.buildTokens(targetCas.getJCas(), "This is a test .\nAnother sentence here ."); - - var layer = AnnotationLayer.builder() // - .withId(1l) // - .forUimaType(targetCas.getTypeSystem().getType(predType.getName())) // - .withType(SpanLayerSupport.TYPE) // - .withAnchoringMode(TOKENS) // - .build(); - var feature = AnnotationFeature.builder() // - .withId(1l) // - .withName("value") // - .withLayer(layer1) // - .build(); - var recommender = Recommender.builder() // - .withId(1l) // - .withEnabled(true) // - .withLayer(layer) // - .withFeature(feature) // - .withMaxRecommendations(3) // - .build(); - - var suggestionCas = createCas(mergeTypeSystems(asList(tsd, createTypeSystemDescription()))); - buildFS(suggestionCas, predType.getName()) // - .withFeature(Annotation._FeatName_begin, 0) // - .withFeature(Annotation._FeatName_end, 4) // - .withFeature("value", "foo") // - .withFeature("value" + FEATURE_NAME_SCORE_SUFFIX, 1.0d) // - .withFeature("value" + FEATURE_NAME_SCORE_EXPLANATION_SUFFIX, "one") // - .withFeature(FEATURE_NAME_IS_PREDICTION, true) // - .buildAndAddToIndexes(); - buildFS(suggestionCas, predType.getName()) // - .withFeature(Annotation._FeatName_begin, 5) // - .withFeature(Annotation._FeatName_end, 12) // - .withFeature("value", "bar") // - .withFeature("value" + FEATURE_NAME_SCORE_SUFFIX, 0.5d) // - .withFeature("value" + FEATURE_NAME_SCORE_EXPLANATION_SUFFIX, "two") // - .withFeature(FEATURE_NAME_IS_PREDICTION, true) // - .buildAndAddToIndexes(); - - var suggestions = extractSuggestions(0, targetCas, suggestionCas, doc1, recommender); - - assertThat(suggestions) // - .extracting( // - AnnotationSuggestion::getLabel, // - AnnotationSuggestion::getScore, // - s -> s.getScoreExplanation().orElse(null), // - AnnotationSuggestion::getPosition) - .containsExactly( // - tuple("foo", 1.0d, "one", new Offset(0, 4)), // - tuple("bar", 0.5d, "two", new Offset(5, 9))); - } - } diff --git a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/SpanSuggestionExtractionTest.java b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/SpanSuggestionExtractionTest.java deleted file mode 100644 index d61e7aee8db..00000000000 --- a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/SpanSuggestionExtractionTest.java +++ /dev/null @@ -1,108 +0,0 @@ -/* - * Licensed to the Technische Universität Darmstadt under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The Technische Universität Darmstadt - * licenses this file to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. - * - * 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.tudarmstadt.ukp.inception.recommendation.service; - -import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_IS_PREDICTION; -import static de.tudarmstadt.ukp.inception.support.uima.AnnotationBuilder.buildAnnotation; -import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.tuple; - -import org.apache.uima.cas.CAS; -import org.apache.uima.fit.factory.CasFactory; -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; - -import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; -import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; -import de.tudarmstadt.ukp.clarin.webanno.model.Project; -import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; -import de.tudarmstadt.ukp.dkpro.core.api.ner.type.NamedEntity; -import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanLayerSupport; -import de.tudarmstadt.ukp.inception.recommendation.api.RecommenderTypeSystemUtils; -import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; -import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; -import de.tudarmstadt.ukp.inception.support.uima.SegmentationUtils; - -class SpanSuggestionExtractionTest -{ - private Project project; - private SourceDocument document; - private CAS originalCas; - - @BeforeEach - void setup() throws Exception - { - project = Project.builder() // - .withId(1l) // - .withName("Test") // - .build(); - document = SourceDocument.builder() // - .withId(1l) // - .withProject(project) // - .withName("Doc") // - .build(); - - originalCas = CasFactory.createCas(); - originalCas.setDocumentText("This is a test."); - - SegmentationUtils.splitSentences(originalCas); - SegmentationUtils.tokenize(originalCas); - } - - @Test - void testSpanExtraction() throws Exception - { - var layer = AnnotationLayer.builder() // - .withId(1l) // - .forJCasClass(NamedEntity.class) // - .withType(SpanLayerSupport.TYPE) // - .build(); - var feature = AnnotationFeature.builder() // - .withLayer(layer) // - .withName(NamedEntity._FeatName_value) // - .build(); - var recommender = Recommender.builder() // - .withId(1l) // - .withName("recommender") // - .withProject(project) // - .withLayer(layer) // - .withFeature(feature) // - .build(); - AnnotationFeature[] aFeatures = { feature }; - - var predictionCas = RecommenderTypeSystemUtils.makePredictionCas(originalCas, aFeatures); - - buildAnnotation(predictionCas, feature.getLayer().getName()) // - .on("\\bis\\b") // - .withFeature(feature.getName(), "verb") // - .withFeature(FEATURE_NAME_IS_PREDICTION, true) // - .buildAndAddToIndexes(); - - var suggestions = SuggestionExtraction.extractSuggestions(1, originalCas, predictionCas, - document, recommender); - - assertThat(suggestions) // - .filteredOn(a -> a instanceof SpanSuggestion) // - .map(a -> (SpanSuggestion) a) // - .extracting( // - SpanSuggestion::getRecommenderName, // - SpanSuggestion::getLabel) // - .containsExactly( // - tuple(recommender.getName(), "verb")); - } -} diff --git a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionExtractionTest.java b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionExtractionTest.java new file mode 100644 index 00000000000..cacd3d87f8c --- /dev/null +++ b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionExtractionTest.java @@ -0,0 +1,275 @@ +/* + * Licensed to the Technische Universität Darmstadt under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The Technische Universität Darmstadt + * licenses this file to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. + * + * 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.tudarmstadt.ukp.inception.recommendation.span; + +import static de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode.CHARACTERS; +import static de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode.SENTENCES; +import static de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode.SINGLE_TOKEN; +import static de.tudarmstadt.ukp.clarin.webanno.model.AnchoringMode.TOKENS; +import static de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService.FEATURE_NAME_IS_PREDICTION; +import static de.tudarmstadt.ukp.inception.recommendation.span.SpanSuggestionSupport.getOffsets; +import static de.tudarmstadt.ukp.inception.recommendation.span.SpanSuggestionSupport.getOffsetsAnchoredOnTokens; +import static de.tudarmstadt.ukp.inception.support.uima.AnnotationBuilder.buildAnnotation; +import static org.apache.uima.fit.factory.JCasFactory.createJCas; +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.tuple; + +import org.apache.uima.cas.CAS; +import org.apache.uima.fit.factory.CasFactory; +import org.apache.uima.fit.factory.JCasFactory; +import org.apache.uima.fit.testing.factory.TokenBuilder; +import org.apache.uima.jcas.tcas.Annotation; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; +import org.springframework.context.ApplicationEventPublisher; + +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationFeature; +import de.tudarmstadt.ukp.clarin.webanno.model.AnnotationLayer; +import de.tudarmstadt.ukp.clarin.webanno.model.Project; +import de.tudarmstadt.ukp.clarin.webanno.model.SourceDocument; +import de.tudarmstadt.ukp.dkpro.core.api.ner.type.NamedEntity; +import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Sentence; +import de.tudarmstadt.ukp.dkpro.core.api.segmentation.type.Token; +import de.tudarmstadt.ukp.inception.annotation.layer.span.SpanLayerSupport; +import de.tudarmstadt.ukp.inception.recommendation.api.LearningRecordService; +import de.tudarmstadt.ukp.inception.recommendation.api.RecommendationService; +import de.tudarmstadt.ukp.inception.recommendation.api.RecommenderTypeSystemUtils; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Offset; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; +import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; +import de.tudarmstadt.ukp.inception.recommendation.api.recommender.ExtractionContext; +import de.tudarmstadt.ukp.inception.recommendation.config.RecommenderProperties; +import de.tudarmstadt.ukp.inception.schema.api.AnnotationSchemaService; +import de.tudarmstadt.ukp.inception.schema.api.feature.FeatureSupportRegistry; +import de.tudarmstadt.ukp.inception.support.uima.SegmentationUtils; + +@ExtendWith(MockitoExtension.class) +class SpanSuggestionExtractionTest +{ + private @Mock RecommendationService recommendationService; + private @Mock LearningRecordService learningRecordService; + private @Mock ApplicationEventPublisher applicationEventPublisher; + private @Mock AnnotationSchemaService schemaService; + private @Mock FeatureSupportRegistry featureSupportRegistry; + private @Mock RecommenderProperties recommenderProperties; + + private TokenBuilder tokenBuilder; + private Project project; + private SourceDocument document; + private CAS originalCas; + + private SpanSuggestionSupport sut; + + @BeforeEach + void setup() throws Exception + { + tokenBuilder = new TokenBuilder<>(Token.class, Sentence.class); + project = Project.builder() // + .withId(1l) // + .withName("Test") // + .build(); + document = SourceDocument.builder() // + .withId(1l) // + .withProject(project) // + .withName("Doc") // + .build(); + + originalCas = CasFactory.createCas(); + originalCas.setDocumentText("This is a test."); + + SegmentationUtils.splitSentences(originalCas); + SegmentationUtils.tokenize(originalCas); + + sut = new SpanSuggestionSupport(recommendationService, learningRecordService, + applicationEventPublisher, schemaService, featureSupportRegistry, + recommenderProperties); + } + + @Test + void testSpanExtraction() throws Exception + { + var layer = AnnotationLayer.builder() // + .withId(1l) // + .forJCasClass(NamedEntity.class) // + .withType(SpanLayerSupport.TYPE) // + .build(); + var feature = AnnotationFeature.builder() // + .withLayer(layer) // + .withName(NamedEntity._FeatName_value) // + .build(); + var recommender = Recommender.builder() // + .withId(1l) // + .withName("recommender") // + .withProject(project) // + .withLayer(layer) // + .withFeature(feature) // + .build(); + AnnotationFeature[] aFeatures = { feature }; + + var predictionCas = RecommenderTypeSystemUtils.makePredictionCas(originalCas, aFeatures); + + buildAnnotation(predictionCas, feature.getLayer().getName()) // + .on("\\bis\\b") // + .withFeature(feature.getName(), "verb") // + .withFeature(FEATURE_NAME_IS_PREDICTION, true) // + .buildAndAddToIndexes(); + + var ctx = new ExtractionContext(0, recommender, document, originalCas, predictionCas); + var suggestions = sut.extractSuggestions(ctx); + + assertThat(suggestions) // + .filteredOn(a -> a instanceof SpanSuggestion) // + .map(a -> (SpanSuggestion) a) // + .extracting( // + SpanSuggestion::getRecommenderName, // + SpanSuggestion::getLabel) // + .containsExactly( // + tuple(recommender.getName(), "verb")); + } + + @Test + void thatZeroWithAnnotationsAreCorrectlyAnchoredOnTokens() throws Exception + { + var jCas = createJCas(); + TokenBuilder.create(Token.class, Sentence.class).buildTokens(jCas, " This is a test. "); + var textLength = jCas.getDocumentText().length(); + var tokens = jCas.select(Token.class).asList(); + var firstTokenBegin = tokens.get(0).getBegin(); + var lastTokenEnd = tokens.get(tokens.size() - 1).getEnd(); + + assertThat(getOffsetsAnchoredOnTokens(jCas.getCas(), new Annotation(jCas, 0, 0))).get() + .as("Zero-width annotation before first token snaps to first token start") + .isEqualTo(new Offset(firstTokenBegin, firstTokenBegin)); + + assertThat(getOffsetsAnchoredOnTokens(jCas.getCas(), + new Annotation(jCas, textLength, textLength))).get() + .as("Zero-width annotation after last token snaps to last token end") + .isEqualTo(new Offset(lastTokenEnd, lastTokenEnd)); + + assertThat(getOffsetsAnchoredOnTokens(jCas.getCas(), new Annotation(jCas, 4, 4))).get() + .as("Zero-width annotation within token remains") // + .isEqualTo(new Offset(4, 4)); + + assertThat(getOffsetsAnchoredOnTokens(jCas.getCas(), new Annotation(jCas, 10, 10))).get() + .as("Zero-width annotation between tokens snaps to end of previous") // + .isEqualTo(new Offset(9, 9)); + } + + @Test + void testOffsetAlignmentWithAnchoringOnCharacters() throws Exception + { + var mode = CHARACTERS; + var targetCas = CasFactory.createCas(); + tokenBuilder.buildTokens(targetCas.getJCas(), "This is a test ."); + + var suggestionCas = JCasFactory.createText(targetCas.getDocumentText()); + + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 1))) // + .as("Trivial case: one character") // + .get().isEqualTo(new Offset(0, 1)); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 4, 8))) // + .as("Leading and trailing space should be removed") // + .get().isEqualTo(new Offset(5, 7)); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, -10, 100))) // + .as("Range should be clipped to document boundaries").get() + .isEqualTo(new Offset(0, targetCas.getDocumentText().length())); + } + + @Test + void testOffsetAlignmentWithAnchoringOnSingleToken() throws Exception + { + var mode = SINGLE_TOKEN; + var targetCas = CasFactory.createCas(); + tokenBuilder.buildTokens(targetCas.getJCas(), "This is a test ."); + + var suggestionCas = JCasFactory.createText(targetCas.getDocumentText()); + + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 1))) // + .as("Reduce to empty if no token is fully covered") // + .isEmpty(); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 4))) // + .as("Trivial case: one token") // + .get().isEqualTo(new Offset(0, 4)); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 7))) // + .as("Discard suggestion if it covers more than one token") // + .isEmpty(); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, -10, 100))) // + .as("Discard suggestion if it covers more than one token (2)") // + .isEmpty(); + } + + @Test + void testOffsetAlignmentWithAnchoringOnTokens() throws Exception + { + var mode = TOKENS; + var targetCas = CasFactory.createCas(); + tokenBuilder.buildTokens(targetCas.getJCas(), "This is a test ."); + + var suggestionCas = JCasFactory.createText(targetCas.getDocumentText()); + + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 1))) // + .as("Reduce to empty if no token is fully covered") // + .isEmpty(); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 4))) // + .as("Trivial case: one token") // + .get().isEqualTo(new Offset(0, 4)); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 7))) // + .as("Trivial case: two tokens") // + .get().isEqualTo(new Offset(0, 7)); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 2, 12))) // + .as("Discard incompletely covered tokens") // + .get().isEqualTo(new Offset(5, 9)); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, -10, 100))) // + .as("Range should be clipped to document boundaries").get() + .isEqualTo(new Offset(0, targetCas.getDocumentText().length())); + } + + @Test + void testOffsetAlignmentWithAnchoringOnSentences() throws Exception + { + var mode = SENTENCES; + var targetCas = CasFactory.createCas(); + tokenBuilder.buildTokens(targetCas.getJCas(), "This is a test .\nAnother sentence here ."); + + var suggestionCas = JCasFactory.createText(targetCas.getDocumentText()); + var sentences = targetCas.select(Sentence.class).asList(); + var sent1 = sentences.get(0); + var sent2 = sentences.get(1); + + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 1))) // + .as("Reduce to empty if no sentence is fully covered") // + .isEmpty(); + assertThat(getOffsets(mode, targetCas, + new Annotation(suggestionCas, sent1.getBegin(), sent1.getEnd()))) // + .as("Trivial case: one sentence") // + .get().isEqualTo(new Offset(sent1.getBegin(), sent1.getEnd())); + assertThat(getOffsets(mode, targetCas, + new Annotation(suggestionCas, sent1.getBegin(), sent2.getEnd()))) // + .as("Trivial case: two sentences") // + .get().isEqualTo(new Offset(sent1.getBegin(), sent2.getEnd())); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, 0, 30))) // + .as("Discard incompletely covered sentences") // + .get().isEqualTo(new Offset(0, 16)); + assertThat(getOffsets(mode, targetCas, new Annotation(suggestionCas, -10, 100))) // + .as("Range should be clipped to document boundaries").get() + .isEqualTo(new Offset(0, targetCas.getDocumentText().length())); + } +} diff --git a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/SpanSuggestionVisibilityCalculationTest.java b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionVisibilityCalculationTest.java similarity index 84% rename from inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/SpanSuggestionVisibilityCalculationTest.java rename to inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionVisibilityCalculationTest.java index 0bfb80cddcb..a608d91ee24 100644 --- a/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/service/SpanSuggestionVisibilityCalculationTest.java +++ b/inception/inception-recommendation/src/test/java/de/tudarmstadt/ukp/inception/recommendation/span/SpanSuggestionVisibilityCalculationTest.java @@ -15,16 +15,16 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package de.tudarmstadt.ukp.inception.recommendation.service; +package de.tudarmstadt.ukp.inception.recommendation.span; import static de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction.REJECTED; import static de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction.SKIPPED; import static de.tudarmstadt.ukp.inception.recommendation.service.Fixtures.getInvisibleSuggestions; import static de.tudarmstadt.ukp.inception.recommendation.service.Fixtures.getVisibleSuggestions; import static de.tudarmstadt.ukp.inception.recommendation.service.Fixtures.makeSpanSuggestionGroup; -import static de.tudarmstadt.ukp.inception.recommendation.service.Fixtures.makeSuggestion; -import static de.tudarmstadt.ukp.inception.recommendation.service.SpanSuggestionSupport.hideSuggestionsRejectedOrSkipped; +import static de.tudarmstadt.ukp.inception.recommendation.span.SpanSuggestionSupport.hideSuggestionsRejectedOrSkipped; import static java.util.Arrays.asList; +import static java.util.Collections.emptyList; import static java.util.stream.Collectors.toList; import static org.apache.uima.cas.CAS.TYPE_NAME_STRING; import static org.assertj.core.api.Assertions.assertThat; @@ -52,7 +52,7 @@ import de.tudarmstadt.ukp.inception.recommendation.api.model.AnnotationSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecord; import de.tudarmstadt.ukp.inception.recommendation.api.model.LearningRecordUserAction; -import de.tudarmstadt.ukp.inception.recommendation.api.model.Offset; +import de.tudarmstadt.ukp.inception.recommendation.api.model.Recommender; import de.tudarmstadt.ukp.inception.recommendation.api.model.SpanSuggestion; import de.tudarmstadt.ukp.inception.recommendation.api.model.SuggestionDocumentGroup; import de.tudarmstadt.ukp.inception.schema.api.AnnotationSchemaService; @@ -94,7 +94,7 @@ public void setUp() throws Exception lenient().when(annoService.listSupportedFeatures(layer)).thenReturn(asList(feature)); lenient().when(annoService.listSupportedFeatures(layer2)).thenReturn(asList(feature2)); - sut = new SpanSuggestionSupport(null, learningRecordService, null, annoService); + sut = new SpanSuggestionSupport(null, learningRecordService, null, annoService, null, null); } @Test @@ -212,30 +212,31 @@ public void thatVisibilityIsRestoredWhenOverlappingAnnotationIsRemoved() throws @Test public void thatOverlappingSuggestionsAreNotHiddenWhenStackingIsEnabled() throws Exception { - doReturn(new ArrayList<>()).when(learningRecordService).listLearningRecords(TEST_USER, - TEST_USER, layer); + doReturn(emptyList()).when(learningRecordService).listLearningRecords(TEST_USER, TEST_USER, + layer); layer.setOverlapMode(OverlapMode.ANY_OVERLAP); + var rec = Recommender.builder().withId(123l).withName("rec").withLayer(layer) + .withFeature(feature).build(); var cas = JCasFactory.createText("a b", "de"); var suggestion1 = SpanSuggestion.builder() // .withId(1) // - .withDocumentName(doc.getName()) // - .withLayerId(layer.getId()) // - .withFeature(NamedEntity._FeatName_value) // + .withDocument(doc) // + .withRecommender(rec) // .withLabel("blah") // - .withPosition(new Offset(0, 1)) // + .withPosition(0, 1) // .build(); var suggestion2 = SpanSuggestion.builder() // .withId(2) // - .withDocumentName(doc.getName()) // - .withLayerId(layer.getId()) // - .withFeature(NamedEntity._FeatName_value) // + .withDocument(doc) // + .withRecommender(rec) // .withLabel("blah") // - .withPosition(new Offset(1, 2)) // + .withPosition(1, 2) // .build(); - var suggestions = new SuggestionDocumentGroup<>(asList(suggestion1, suggestion2)); + var suggestions = SuggestionDocumentGroup.groupsOfType(SpanSuggestion.class, + asList(suggestion1, suggestion2)); sut.calculateSuggestionVisibility(TEST_USER, doc, cas.getCas(), TEST_USER, layer, suggestions, 0, 2); @@ -278,17 +279,23 @@ public void thatOverlappingSuggestionsAreNotHiddenWhenStackingIsEnabled() throws @Test void thatRejectedSuggestionIsHidden() { + var rec1 = Recommender.builder().withId(1l).withLayer(layer).withFeature(feature).build(); + var rec2 = Recommender.builder().withId(1l).withLayer(layer2).withFeature(feature).build(); + var rec3 = Recommender.builder().withId(1l).withLayer(layer).withFeature(feature2).build(); + var label = "x"; + var records = asList(LearningRecord.builder() // .withSourceDocument(doc) // .withLayer(layer) // .withAnnotationFeature(feature) // .withOffsetBegin(0) // .withOffsetEnd(10) // - .withAnnotation("x") // + .withAnnotation(label) // .withUserAction(REJECTED) // .build()); - var docSuggestion = makeSuggestion(0, 10, "x", doc, layer, feature); + var docSuggestion = SpanSuggestion.builder().withRecommender(rec1).withDocument(doc) + .withLabel(label).withPosition(0, 10).build(); assertThat(docSuggestion.isVisible()).isTrue(); hideSuggestionsRejectedOrSkipped(docSuggestion, records); assertThat(docSuggestion.isVisible()) // @@ -296,21 +303,24 @@ void thatRejectedSuggestionIsHidden() .isFalse(); assertThat(docSuggestion.getReasonForHiding().trim()).isEqualTo("rejected"); - var doc2Suggestion = makeSuggestion(0, 10, "x", doc2, layer, feature); + var doc2Suggestion = SpanSuggestion.builder().withRecommender(rec1).withDocument(doc2) + .withLabel(label).withPosition(0, 10).build(); assertThat(doc2Suggestion.isVisible()).isTrue(); hideSuggestionsRejectedOrSkipped(doc2Suggestion, records); assertThat(doc2Suggestion.isVisible()) // .as("Suggestion in other document should not be hidden") // .isTrue(); - var doc3Suggestion = makeSuggestion(0, 10, "x", doc, layer2, feature); + var doc3Suggestion = SpanSuggestion.builder().withRecommender(rec2).withDocument(doc) + .withLabel(label).withPosition(0, 10).build(); assertThat(doc3Suggestion.isVisible()).isTrue(); hideSuggestionsRejectedOrSkipped(doc3Suggestion, records); assertThat(doc3Suggestion.isVisible()) // .as("Suggestion in other layer should not be hidden") // .isTrue(); - var doc4Suggestion = makeSuggestion(0, 10, "x", doc, layer, feature2); + var doc4Suggestion = SpanSuggestion.builder().withRecommender(rec3).withDocument(doc) + .withLabel(label).withPosition(0, 10).build(); assertThat(doc4Suggestion.isVisible()).isTrue(); hideSuggestionsRejectedOrSkipped(doc3Suggestion, records); assertThat(doc4Suggestion.isVisible()) // @@ -321,17 +331,23 @@ void thatRejectedSuggestionIsHidden() @Test void thatSkippedSuggestionIsHidden() { + var rec1 = Recommender.builder().withId(1l).withLayer(layer).withFeature(feature).build(); + var rec2 = Recommender.builder().withId(1l).withLayer(layer2).withFeature(feature).build(); + var rec3 = Recommender.builder().withId(1l).withLayer(layer).withFeature(feature2).build(); + var label = "x"; + var records = asList(LearningRecord.builder() // .withSourceDocument(doc) // .withLayer(layer) // .withAnnotationFeature(feature) // .withOffsetBegin(0) // .withOffsetEnd(10) // - .withAnnotation("x") // + .withAnnotation(label) // .withUserAction(SKIPPED) // .build()); - var docSuggestion = makeSuggestion(0, 10, "x", doc, layer, feature); + var docSuggestion = SpanSuggestion.builder().withRecommender(rec1).withDocument(doc) + .withLabel(label).withPosition(0, 10).build(); assertThat(docSuggestion.isVisible()).isTrue(); hideSuggestionsRejectedOrSkipped(docSuggestion, records); assertThat(docSuggestion.isVisible()) // @@ -339,21 +355,23 @@ void thatSkippedSuggestionIsHidden() .isFalse(); assertThat(docSuggestion.getReasonForHiding().trim()).isEqualTo("skipped"); - var doc2Suggestion = makeSuggestion(0, 10, "x", doc2, layer, feature); + var doc2Suggestion = SpanSuggestion.builder().withRecommender(rec1).withDocument(doc2) + .withLabel(label).withPosition(0, 10).build(); assertThat(doc2Suggestion.isVisible()).isTrue(); hideSuggestionsRejectedOrSkipped(doc2Suggestion, records); assertThat(doc2Suggestion.isVisible()) // .as("Suggestion in other document should not be hidden") // .isTrue(); - var doc3Suggestion = makeSuggestion(0, 10, "x", doc, layer2, feature); - assertThat(doc3Suggestion.isVisible()).isTrue(); + var doc3Suggestion = SpanSuggestion.builder().withRecommender(rec2).withDocument(doc) + .withLabel(label).withPosition(0, 10).build(); hideSuggestionsRejectedOrSkipped(doc3Suggestion, records); assertThat(doc3Suggestion.isVisible()) // .as("Suggestion in other layer should not be hidden") // .isTrue(); - var doc4Suggestion = makeSuggestion(0, 10, "x", doc, layer, feature2); + var doc4Suggestion = SpanSuggestion.builder().withRecommender(rec3).withDocument(doc) + .withLabel(label).withPosition(0, 10).build(); assertThat(doc4Suggestion.isVisible()).isTrue(); hideSuggestionsRejectedOrSkipped(doc3Suggestion, records); assertThat(doc4Suggestion.isVisible()) //