mirror of https://github.com/apache/lucene.git
SOLR-15149: model creation errors fixes (#2350)
SOLR-15149: model loading errors fix + tests
This commit is contained in:
parent
f993c392f1
commit
227ef3b397
|
@ -239,6 +239,8 @@ Optimizations
|
||||||
Bug Fixes
|
Bug Fixes
|
||||||
---------------------
|
---------------------
|
||||||
* SOLR-15078: Fix ExpandComponent behavior when expanding on numeric fields to differentiate '0' group from null group (hossman)
|
* SOLR-15078: Fix ExpandComponent behavior when expanding on numeric fields to differentiate '0' group from null group (hossman)
|
||||||
|
* SOLR-15149: Better exception handling for LTR model creation errors (Alessandro Benedetti, Christine Poerschke)
|
||||||
|
|
||||||
|
|
||||||
Other Changes
|
Other Changes
|
||||||
---------------------
|
---------------------
|
||||||
|
|
|
@ -108,7 +108,7 @@ public abstract class LTRScoringModel implements Accountable {
|
||||||
SolrPluginUtils.invokeSetters(model, params.entrySet());
|
SolrPluginUtils.invokeSetters(model, params.entrySet());
|
||||||
}
|
}
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
throw new ModelException("Model type does not exist " + className, e);
|
throw new ModelException("Model loading failed for " + className, e);
|
||||||
}
|
}
|
||||||
model.validate();
|
model.validate();
|
||||||
return model;
|
return model;
|
||||||
|
|
|
@ -80,10 +80,10 @@ public class LinearModel extends LTRScoringModel {
|
||||||
|
|
||||||
public void setWeights(Object weights) {
|
public void setWeights(Object weights) {
|
||||||
@SuppressWarnings({"unchecked"})
|
@SuppressWarnings({"unchecked"})
|
||||||
final Map<String,Double> modelWeights = (Map<String,Double>) weights;
|
final Map<String,Number> modelWeights = (Map<String, Number>) weights;
|
||||||
for (int ii = 0; ii < features.size(); ++ii) {
|
for (int ii = 0; ii < features.size(); ++ii) {
|
||||||
final String key = features.get(ii).getName();
|
final String key = features.get(ii).getName();
|
||||||
final Double val = modelWeights.get(key);
|
final Number val = modelWeights.get(key);
|
||||||
featureToWeight[ii] = (val == null ? null : val.floatValue());
|
featureToWeight[ii] = (val == null ? null : val.floatValue());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -294,11 +294,18 @@ public class ManagedModelStore extends ManagedResource implements ManagedResourc
|
||||||
return modelMap;
|
return modelMap;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Feature lookupFeatureFromFeatureMap(Map<String,Object> featureMap,
|
private static Feature lookupFeatureFromFeatureMap(Map<String, Object> featureMap, FeatureStore featureStore)
|
||||||
FeatureStore featureStore) {
|
{
|
||||||
final String featureName = (String)featureMap.get(NAME_KEY);
|
final String featureName = (String) featureMap.get(NAME_KEY);
|
||||||
return (featureName == null ? null
|
Feature extractedFromStore = featureName == null ? null : featureStore.get(featureName);
|
||||||
: featureStore.get(featureName));
|
if (extractedFromStore == null) {
|
||||||
|
if (featureStore.getFeatures().isEmpty()) {
|
||||||
|
throw new ModelException("Missing or empty feature store: " + featureStore.getName());
|
||||||
|
} else {
|
||||||
|
throw new ModelException("Feature: " + featureName + " not found in store: " + featureStore.getName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return extractedFromStore;
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
|
|
|
@ -0,0 +1,19 @@
|
||||||
|
{
|
||||||
|
"class": "org.apache.solr.ltr.model.LinearModel",
|
||||||
|
"name": "6029760550880411648",
|
||||||
|
"store": "test",
|
||||||
|
"features": [
|
||||||
|
{
|
||||||
|
"name": "notExist1"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "notExist2"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"params": {
|
||||||
|
"weights": {
|
||||||
|
"notExist1": 0.0000000000,
|
||||||
|
"notExist2": 0.1000000000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,46 @@
|
||||||
|
{
|
||||||
|
"class": "org.apache.solr.ltr.model.LinearModel",
|
||||||
|
"name": "6029760550880411648",
|
||||||
|
"store": "not_existent_store",
|
||||||
|
"features": [
|
||||||
|
{
|
||||||
|
"name": "title"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "description"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "keywords"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "popularity",
|
||||||
|
"norm": {
|
||||||
|
"class": "org.apache.solr.ltr.norm.MinMaxNormalizer",
|
||||||
|
"params": {
|
||||||
|
"min": "0.0f",
|
||||||
|
"max": "10.0f"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "text"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "queryIntentPerson"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "queryIntentCompany"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"params": {
|
||||||
|
"weights": {
|
||||||
|
"title": 0.0000000000,
|
||||||
|
"description": 0.1000000000,
|
||||||
|
"keywords": 0.2000000000,
|
||||||
|
"popularity": 0.3000000000,
|
||||||
|
"text": 0.4000000000,
|
||||||
|
"queryIntentPerson": 0.1231231,
|
||||||
|
"queryIntentCompany": 0.12121211
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,39 @@
|
||||||
|
{
|
||||||
|
"class":"org.apache.solr.ltr.model.MultipleAdditiveTreesModel",
|
||||||
|
"name":"multipleadditivetreesmodel",
|
||||||
|
"store": "not_existent_store",
|
||||||
|
"features":[
|
||||||
|
{ "name": "matchedTitle"},
|
||||||
|
{ "name": "constantScoreToForceMultipleAdditiveTreesScoreAllDocs"}
|
||||||
|
],
|
||||||
|
"params":{
|
||||||
|
"trees": [
|
||||||
|
{
|
||||||
|
"weight" : "1f",
|
||||||
|
"root": {
|
||||||
|
"feature": "matchedTitle",
|
||||||
|
"threshold": "0.5f",
|
||||||
|
"left" : {
|
||||||
|
"value" : "-100"
|
||||||
|
},
|
||||||
|
"right": {
|
||||||
|
"feature" : "constantScoreToForceMultipleAdditiveTreesScoreAllDocs",
|
||||||
|
"threshold": "10.0f",
|
||||||
|
"left" : {
|
||||||
|
"value" : "50"
|
||||||
|
},
|
||||||
|
"right" : {
|
||||||
|
"value" : "75"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"weight" : "2f",
|
||||||
|
"root": {
|
||||||
|
"value" : "-10"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,38 @@
|
||||||
|
{
|
||||||
|
"class":"org.apache.solr.ltr.model.MultipleAdditiveTreesModel",
|
||||||
|
"name":"multipleadditivetreesmodel",
|
||||||
|
"features":[
|
||||||
|
{ "name": "notExist1"},
|
||||||
|
{ "name": "notExist2"}
|
||||||
|
],
|
||||||
|
"params":{
|
||||||
|
"trees": [
|
||||||
|
{
|
||||||
|
"weight" : "1f",
|
||||||
|
"root": {
|
||||||
|
"feature": "notExist1",
|
||||||
|
"threshold": "0.5f",
|
||||||
|
"left" : {
|
||||||
|
"value" : "-100"
|
||||||
|
},
|
||||||
|
"right": {
|
||||||
|
"feature" : "notExist2",
|
||||||
|
"threshold": "10.0f",
|
||||||
|
"left" : {
|
||||||
|
"value" : "50"
|
||||||
|
},
|
||||||
|
"right" : {
|
||||||
|
"value" : "75"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"weight" : "2f",
|
||||||
|
"root": {
|
||||||
|
"value" : "-10"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
|
@ -184,6 +184,29 @@ public class TestLinearModel extends TestRerankBase {
|
||||||
assertEquals(expectedException.toString(), ex.toString());
|
assertEquals(expectedException.toString(), ex.toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void integerAndLongFeatureWeights_shouldCreateModel() {
|
||||||
|
final List<Feature> features = getFeatures(new String[]
|
||||||
|
{"constant1", "constant5"});
|
||||||
|
final List<Normalizer> norms =
|
||||||
|
new ArrayList<>(Collections.nCopies(features.size(),IdentityNormalizer.INSTANCE));
|
||||||
|
|
||||||
|
final Map<String,Object> weights = new HashMap<>();
|
||||||
|
weights.put("constant1", 100L);
|
||||||
|
weights.put("constant5", 1);
|
||||||
|
|
||||||
|
Map<String,Object> params = new HashMap<>();
|
||||||
|
params.put("weights", weights);
|
||||||
|
|
||||||
|
final LTRScoringModel ltrScoringModel = createLinearModel("test6",
|
||||||
|
features, norms, "test", fstore.getFeatures(),
|
||||||
|
params);
|
||||||
|
|
||||||
|
store.addModel(ltrScoringModel);
|
||||||
|
final LTRScoringModel m = store.getModel("test6");
|
||||||
|
assertEquals(ltrScoringModel, m);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void emptyFeaturesTest() {
|
public void emptyFeaturesTest() {
|
||||||
final ModelException expectedException =
|
final ModelException expectedException =
|
||||||
|
@ -206,4 +229,28 @@ public class TestLinearModel extends TestRerankBase {
|
||||||
assertEquals(expectedException.toString(), ex.toString());
|
assertEquals(expectedException.toString(), ex.toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void notExistentStore_shouldThrowMeaningFulException(){
|
||||||
|
final ModelException expectedException =
|
||||||
|
new ModelException("Feature Store not found: not_existent_store");
|
||||||
|
|
||||||
|
ModelException ex = expectThrows(ModelException.class, () -> {
|
||||||
|
createModelFromFiles("linear-model_notExistentStore.json",
|
||||||
|
"features-store-test-model.json");
|
||||||
|
});
|
||||||
|
assertEquals(expectedException.toString(), ex.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void notExistentFeature_shouldThrowMeaningFulException(){
|
||||||
|
final ModelException expectedException =
|
||||||
|
new ModelException("Feature:notExist1 not found in store: test");
|
||||||
|
|
||||||
|
ModelException ex = expectThrows(ModelException.class, () -> {
|
||||||
|
createModelFromFiles("linear-model_notExistentFeature.json",
|
||||||
|
"features-store-test-model.json");
|
||||||
|
});
|
||||||
|
assertEquals(expectedException.toString(), ex.toString());
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -225,4 +225,29 @@ public class TestMultipleAdditiveTreesModel extends TestRerankBase {
|
||||||
});
|
});
|
||||||
assertEquals(expectedException.toString(), ex.toString());
|
assertEquals(expectedException.toString(), ex.toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void multipleAdditiveTreesTestMissingFeatureStore(){
|
||||||
|
final ModelException expectedException =
|
||||||
|
new ModelException("Feature Store not found: not_existent_store");
|
||||||
|
|
||||||
|
ModelException ex = expectThrows(ModelException.class, () -> {
|
||||||
|
createModelFromFiles("multipleadditivetreesmodel_notExistentStore.json",
|
||||||
|
"multipleadditivetreesmodel_features.json");
|
||||||
|
});
|
||||||
|
assertEquals(expectedException.toString(), ex.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void multipleAdditiveTreesTestUnknownFeature(){
|
||||||
|
final ModelException expectedException =
|
||||||
|
new ModelException("Feature:notExist1 not found in store: _DEFAULT_");
|
||||||
|
|
||||||
|
ModelException ex = expectThrows(ModelException.class, () -> {
|
||||||
|
createModelFromFiles("multipleadditivetreesmodel_unknownFeature.json",
|
||||||
|
"multipleadditivetreesmodel_features.json");
|
||||||
|
});
|
||||||
|
assertEquals(expectedException.toString(), ex.toString());
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue