Search in sources :

Example 1 with ValidationFailure

use of org.mongodb.morphia.query.validation.ValidationFailure in project morphia by mongodb.

the class QueryValidator method validateQuery.

/**
     * Validate the path, and value type, returning the mapped field for the field at the path
     */
static MappedField validateQuery(final Class clazz, final Mapper mapper, final StringBuilder origProp, final FilterOperator op, final Object val, final boolean validateNames, final boolean validateTypes) {
    MappedField mf = null;
    final String prop = origProp.toString();
    boolean hasTranslations = false;
    if (!origProp.substring(0, 1).equals("$")) {
        final String[] parts = prop.split("\\.");
        if (clazz == null) {
            return null;
        }
        MappedClass mc = mapper.getMappedClass(clazz);
        //CHECKSTYLE:OFF
        for (int i = 0; ; ) {
            //CHECKSTYLE:ON
            final String part = parts[i];
            boolean fieldIsArrayOperator = part.equals("$") || part.matches("[0-9]+");
            mf = mc.getMappedField(part);
            //translate from java field name to stored field name
            if (mf == null && !fieldIsArrayOperator) {
                mf = mc.getMappedFieldByJavaField(part);
                if (validateNames && mf == null) {
                    throw new ValidationException(format("The field '%s' could not be found in '%s' while validating - %s; if " + "you wish to continue please disable validation.", part, mc.getClazz().getName(), prop));
                }
                hasTranslations = true;
                if (mf != null) {
                    parts[i] = mf.getNameToStore();
                }
            }
            i++;
            if (mf != null && mf.isMap()) {
                //skip the map key validation, and move to the next part
                i++;
            }
            if (i >= parts.length) {
                break;
            }
            if (!fieldIsArrayOperator) {
                //catch people trying to search/update into @Reference/@Serialized fields
                if (validateNames && !canQueryPast(mf)) {
                    throw new ValidationException(format("Cannot use dot-notation past '%s' in '%s'; found while" + " validating - %s", part, mc.getClazz().getName(), prop));
                }
                if (mf == null && (mc.isInterface() || !validateNames)) {
                    break;
                } else if (mf == null) {
                    throw new ValidationException(format("The field '%s' could not be found in '%s'", prop, mc.getClazz().getName()));
                }
                //get the next MappedClass for the next field validation
                mc = mapper.getMappedClass((mf.isSingleValue()) ? mf.getType() : mf.getSubClass());
            }
        }
        //record new property string if there has been a translation to any part
        if (hasTranslations) {
            // clear existing content
            origProp.setLength(0);
            origProp.append(parts[0]);
            for (int i = 1; i < parts.length; i++) {
                origProp.append('.');
                origProp.append(parts[i]);
            }
        }
        if (validateTypes && mf != null) {
            List<ValidationFailure> typeValidationFailures = new ArrayList<ValidationFailure>();
            boolean compatibleForType = isCompatibleForOperator(mc, mf, mf.getType(), op, val, typeValidationFailures);
            List<ValidationFailure> subclassValidationFailures = new ArrayList<ValidationFailure>();
            boolean compatibleForSubclass = isCompatibleForOperator(mc, mf, mf.getSubClass(), op, val, subclassValidationFailures);
            if ((mf.isSingleValue() && !compatibleForType) || mf.isMultipleValues() && !(compatibleForSubclass || compatibleForType)) {
                if (LOG.isWarningEnabled()) {
                    LOG.warning(format("The type(s) for the query/update may be inconsistent; using an instance of type '%s' " + "for the field '%s.%s' which is declared as '%s'", val.getClass().getName(), mf.getDeclaringClass().getName(), mf.getJavaFieldName(), mf.getType().getName()));
                    typeValidationFailures.addAll(subclassValidationFailures);
                    LOG.warning("Validation warnings: \n" + typeValidationFailures);
                }
            }
        }
    }
    return mf;
}
Also used : MappedField(org.mongodb.morphia.mapping.MappedField) ArrayList(java.util.ArrayList) MappedClass(org.mongodb.morphia.mapping.MappedClass) ValidationFailure(org.mongodb.morphia.query.validation.ValidationFailure)

Example 2 with ValidationFailure

use of org.mongodb.morphia.query.validation.ValidationFailure in project morphia by mongodb.

the class QueryForSubtypeTest method testIntSizeShouldBeCompatibleWithArrayList.

@Test
public void testIntSizeShouldBeCompatibleWithArrayList() {
    MappedField attributes = jobMappedClass.getMappedField("attributes");
    boolean compatible = isCompatibleForOperator(jobMappedClass, attributes, ArrayList.class, SIZE, 2, new ArrayList<ValidationFailure>());
    assertThat(compatible, is(true));
}
Also used : MappedField(org.mongodb.morphia.mapping.MappedField) ValidationFailure(org.mongodb.morphia.query.validation.ValidationFailure) Test(org.junit.Test)

Example 3 with ValidationFailure

use of org.mongodb.morphia.query.validation.ValidationFailure in project morphia by mongodb.

the class QueryForSubtypeTest method testImplementingClassIsCompatibleWithInterface.

@Test
public void testImplementingClassIsCompatibleWithInterface() {
    MappedField user = jobMappedClass.getMappedField("owner");
    boolean compatible = isCompatibleForOperator(jobMappedClass, user, User.class, EQUAL, new UserImpl(), new ArrayList<ValidationFailure>());
    assertThat(compatible, is(true));
}
Also used : MappedField(org.mongodb.morphia.mapping.MappedField) ValidationFailure(org.mongodb.morphia.query.validation.ValidationFailure) Test(org.junit.Test)

Example 4 with ValidationFailure

use of org.mongodb.morphia.query.validation.ValidationFailure in project morphia by mongodb.

the class QueryForSubtypeTest method testSubclassOfKeyShouldBeCompatibleWithFieldUser.

@Test
public void testSubclassOfKeyShouldBeCompatibleWithFieldUser() {
    MappedField user = jobMappedClass.getMappedField("owner");
    Key<User> anonymousKeySubclass = new Key<User>(User.class, "User", 212L) {
    };
    boolean compatible = isCompatibleForOperator(jobMappedClass, user, User.class, EQUAL, anonymousKeySubclass, new ArrayList<ValidationFailure>());
    assertThat(compatible, is(true));
}
Also used : MappedField(org.mongodb.morphia.mapping.MappedField) Key(org.mongodb.morphia.Key) ValidationFailure(org.mongodb.morphia.query.validation.ValidationFailure) Test(org.junit.Test)

Aggregations

MappedField (org.mongodb.morphia.mapping.MappedField)4 ValidationFailure (org.mongodb.morphia.query.validation.ValidationFailure)4 Test (org.junit.Test)3 ArrayList (java.util.ArrayList)1 Key (org.mongodb.morphia.Key)1 MappedClass (org.mongodb.morphia.mapping.MappedClass)1