From 8365fcc3a5b2285fc1fe442d6f2eb8a90cbbab3a Mon Sep 17 00:00:00 2001 From: "Brian S. O'Neill" <bronee@gmail.com> Date: Wed, 30 Aug 2006 01:50:06 +0000 Subject: Add standard capabilities --- .../carbonado/constraint/ConstraintDefinition.java | 111 ++++++++++ .../carbonado/constraint/FloatConstraint.java | 201 ++++++++++++++++++ .../carbonado/constraint/IntegerConstraint.java | 233 +++++++++++++++++++++ .../carbonado/constraint/LengthConstraint.java | 173 +++++++++++++++ .../carbonado/constraint/TextConstraint.java | 152 ++++++++++++++ .../amazon/carbonado/constraint/package-info.java | 24 +++ 6 files changed, 894 insertions(+) create mode 100644 src/main/java/com/amazon/carbonado/constraint/ConstraintDefinition.java create mode 100644 src/main/java/com/amazon/carbonado/constraint/FloatConstraint.java create mode 100644 src/main/java/com/amazon/carbonado/constraint/IntegerConstraint.java create mode 100644 src/main/java/com/amazon/carbonado/constraint/LengthConstraint.java create mode 100644 src/main/java/com/amazon/carbonado/constraint/TextConstraint.java create mode 100644 src/main/java/com/amazon/carbonado/constraint/package-info.java (limited to 'src') diff --git a/src/main/java/com/amazon/carbonado/constraint/ConstraintDefinition.java b/src/main/java/com/amazon/carbonado/constraint/ConstraintDefinition.java new file mode 100644 index 0000000..22514fc --- /dev/null +++ b/src/main/java/com/amazon/carbonado/constraint/ConstraintDefinition.java @@ -0,0 +1,111 @@ +/* + * Copyright 2006 Amazon Technologies, Inc. or its affiliates. + * Amazon, Amazon.com and Carbonado are trademarks or registered trademarks + * of Amazon Technologies, Inc. or its affiliates. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.amazon.carbonado.constraint; + +import java.lang.annotation.*; + +/** + * Allows annotations to be defined that restrict property values. The + * annotation is just a pointer to a constraint checking class. If the + * constraint class is not explicitly provided, it defaults to a static inner + * class named "Constraint" in the annotation itself. + * + * <p>The constraint class must have a public constructor that accepts the + * annotation that has the ConstraintDefinition annotation. It must also define + * several "constrain" methods which perform constraint checks on specific + * property types. + * <p> + * Example integer constraint: + * <pre> + * @Documented + * <b>@Retention(RetentionPolicy.RUNTIME)</b> + * <b>@Target(ElementType.METHOD)</b> + * <b>@ConstraintDefinition</b> + * public @interface IntegerConstraint { + * int min() default Integer.MIN_VALUE; + * + * int max() default Integer.MAX_VALUE; + * + * public static class Constraint { + * private final String propertyName; + * private final int min; + * private final int max; + * + * // Constructor may throw a MalformedTypeException if + * // params supplied by annotation are illegal. + * + * /** + * * @param type optional type of object that contains the constrained property + * * @param propertyName name of property with constraint + * * @param annotation specific annotation that binds to this constraint class + * */ + * public Constraint(Class type, String propertyName, IntegerConstraint annotation) { + * this.propertyName = propertyName; + * this.min = annotation.min(); + * this.max = annotation.max(); + * } + * + * // Define a constrain method for each supported property type. + * + * /** + * * @param propertyValue specific value to constrain + * */ + * public void constrain(int propertyValue) throws IllegalArgumentException { + * if (propertyValue < min || propertyValue > max) { + * throw new IllegalArgumentException + * ("Value for \"" + propertyName + "\" must be in range " + + * min + ".." + max + ": " + propertyValue); + * } + * } + * } + * } + * </pre> + * + * The newly defined integer constraint can be applied to property mutators. + * + * <pre> + * public interface UserInfo extends Storable { + * ... + * + * // Constraint is called before setting age. + * <b>@IntegerConstraint(min=0, max=120)</b> + * void setAge(int value); + * } + * </pre> + * + * @author Brian S O'Neill + */ +@Documented +@Retention(RetentionPolicy.RUNTIME) +@Target(ElementType.ANNOTATION_TYPE) +public @interface ConstraintDefinition { + /** + * Specify class which will perform constraint checking. Must have a public + * constructor with the signature + * <code>(Class type, String propertyName, <i>Annotation</i>)</code>, + * where <code><i>Annotation</i></code> refers to the annotation with the + * constraint definition. + * + * <p>The implementation class need not be explicitly specified. By + * default, the constraint class must be a static inner class of the + * annotation, named "Constraint". + */ + // Use void.class to mean default. + Class implementation() default void.class; +} diff --git a/src/main/java/com/amazon/carbonado/constraint/FloatConstraint.java b/src/main/java/com/amazon/carbonado/constraint/FloatConstraint.java new file mode 100644 index 0000000..31b6503 --- /dev/null +++ b/src/main/java/com/amazon/carbonado/constraint/FloatConstraint.java @@ -0,0 +1,201 @@ +/* + * Copyright 2006 Amazon Technologies, Inc. or its affiliates. + * Amazon, Amazon.com and Carbonado are trademarks or registered trademarks + * of Amazon Technologies, Inc. or its affiliates. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.amazon.carbonado.constraint; + +import java.lang.annotation.*; +import java.util.Arrays; + +import com.amazon.carbonado.MalformedTypeException; + +/** + * Limits the value of a property to be a member of a specific set. The + * property value may be a boxed or unboxed float, double, String, + * CharSequence, char, Character, or character array. If the property value is + * outside the set, an IllegalArgumentException is thrown. + * + * <p>Example:<pre> + * public interface PolarCoordinate extends Storable { + * double getTheta(); + * + * <b>@FloatConstraint(min=0, max=Math.PI * 2, disallowed=Double.NaN)</b> + * void setTheta(double radians); + * + * ... + * } + * </pre> + * + * @author Brian S O'Neill + * @see IntegerConstraint + * @see TextConstraint + */ +@Documented +@Retention(RetentionPolicy.RUNTIME) +@Target(ElementType.METHOD) +@ConstraintDefinition +public @interface FloatConstraint { + /** + * Specific allowed values for property. Default is unlimited. + */ + double[] allowed() default {}; + + /** + * Specific disallowed values for property. Default is none. + */ + double[] disallowed() default {}; + + /** + * Specify minimum allowed value for float/double property. Default is unlimited. + */ + double min() default Double.MIN_VALUE; + + /** + * Specify maximum allowed value for float/double property. Default is unlimited. + */ + double max() default Double.MAX_VALUE; + + /** + * Constraint implementation for {@link FloatConstraint}. + */ + public static class Constraint { + private final String mPropertyName; + private final double mMinValue; + private final double mMaxValue; + + /** Disallowed values, sorted for binary search. */ + private final double[] mDisallowed; + + /** Allowed values, sorted for binary search. */ + private final double[] mAllowed; + + /** + * @param type type of object that contains the constrained property + * @param propertyName name of property with constraint + * @param ann specific annotation that binds to this constraint class + */ + public Constraint(Class<?> type, String propertyName, FloatConstraint ann) { + this(type, propertyName, + ann.min(), ann.max(), ann.allowed(), ann.disallowed()); + } + + /** + * @param type type of object that contains the constrained property + * @param propertyName name of property with constraint + * @param min minimum allowed value + * @param max maximum allowed value + * @param allowed optional set of allowed values + * @param disallowed optional set of disallowed values + */ + public Constraint(Class<?> type, String propertyName, + double min, double max, double[] allowed, double[] disallowed) { + mPropertyName = propertyName; + mMinValue = min; + mMaxValue = max; + if (mMaxValue < mMinValue) { + throw new MalformedTypeException + (type, "Illegal range for float constraint on property \"" + + propertyName + "\": " + rangeString()); + } + + if (disallowed == null || disallowed.length == 0) { + disallowed = null; + } else { + disallowed = disallowed.clone(); + Arrays.sort(disallowed); + } + + if (allowed == null || allowed.length == 0) { + allowed = null; + } else { + allowed = allowed.clone(); + Arrays.sort(allowed); + for (double value : allowed) { + if (value < mMinValue || value > mMaxValue || + (disallowed != null && Arrays.binarySearch(disallowed, value) >= 0)) { + throw new MalformedTypeException + (type, "Allowed value contradiction for float constraint " + + "on property \"" + propertyName + "\": " + value); + } + } + + // No need to have a set of disallowed values. + disallowed = null; + } + + mDisallowed = disallowed; + mAllowed = allowed; + } + + public void constrain(double propertyValue) { + if (propertyValue < mMinValue || propertyValue > mMaxValue) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" must be in range " + + rangeString() + ": " + propertyValue); + } + if (mDisallowed != null && Arrays.binarySearch(mDisallowed, propertyValue) >= 0) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is disallowed: " + propertyValue); + } + if (mAllowed != null && Arrays.binarySearch(mAllowed, propertyValue) < 0) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is not allowed: " + propertyValue); + } + } + + public void constrain(CharSequence propertyValue) { + if (propertyValue != null) { + try { + constrain(Double.parseDouble(propertyValue.toString())); + } catch (NumberFormatException e) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is not a number: " + + propertyValue); + } + } + } + + public void constrain(char propertyValue) { + if ('0' <= propertyValue && propertyValue <= '9') { + constrain((double) (propertyValue - '0')); + } else { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is not a number: " + propertyValue); + } + } + + public void constrain(char[] propertyValue) { + if (propertyValue != null) { + constrain(new String(propertyValue)); + } + } + + private String rangeString() { + StringBuilder b = new StringBuilder(); + b.append('('); + if (mMinValue > Double.MIN_VALUE) { + b.append(mMinValue); + } + b.append(".."); + if (mMaxValue < Double.MAX_VALUE) { + b.append(mMaxValue); + } + b.append(')'); + return b.toString(); + } + } +} diff --git a/src/main/java/com/amazon/carbonado/constraint/IntegerConstraint.java b/src/main/java/com/amazon/carbonado/constraint/IntegerConstraint.java new file mode 100644 index 0000000..69fa5cc --- /dev/null +++ b/src/main/java/com/amazon/carbonado/constraint/IntegerConstraint.java @@ -0,0 +1,233 @@ +/* + * Copyright 2006 Amazon Technologies, Inc. or its affiliates. + * Amazon, Amazon.com and Carbonado are trademarks or registered trademarks + * of Amazon Technologies, Inc. or its affiliates. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.amazon.carbonado.constraint; + +import java.lang.annotation.*; +import java.util.Arrays; + +import com.amazon.carbonado.MalformedTypeException; + +/** + * Limits the value of a property to be a member of a specific set. The + * property value may be a boxed or unboxed byte, short, int, long, float, + * double, String, CharSequence, char, Character, or character array. If the + * property value is outside the set, an IllegalArgumentException is thrown. + * + * <p>Example:<pre> + * public interface UserInfo extends Storable { + * int getAge(); + * + * <b>@IntegerConstraint(min=0, max=120)</b> + * void setAge(int value); + * + * int getRoleID(); + * + * <b>@IntegerConstraint(allowed={ROLE_REGULAR, ROLE_ADMIN})</b> + * void setRoleID(int role); + * + * ... + * } + * </pre> + * + * @author Brian S O'Neill + * @see FloatConstraint + * @see TextConstraint + */ +@Documented +@Retention(RetentionPolicy.RUNTIME) +@Target(ElementType.METHOD) +@ConstraintDefinition +public @interface IntegerConstraint { + /** + * Specific allowed values for property. Default is unlimited. + */ + long[] allowed() default {}; + + /** + * Specific disallowed values for property. Default is none. + */ + long[] disallowed() default {}; + + /** + * Specify minimum allowed value for integer property. Default is unlimited. + */ + long min() default Long.MIN_VALUE; + + /** + * Specify maximum allowed value for integer property. Default is unlimited. + */ + long max() default Long.MAX_VALUE; + + /** + * Constraint implementation for {@link IntegerConstraint}. + */ + public static class Constraint { + private final String mPropertyName; + + private final long mMinValue; + private final long mMaxValue; + + /** Disallowed values, sorted for binary search. */ + private final long[] mDisallowed; + + /** Allowed values, sorted for binary search. */ + private final long[] mAllowed; + + /** + * @param type type of object that contains the constrained property + * @param propertyName name of property with constraint + * @param ann specific annotation that binds to this constraint class + */ + public Constraint(Class<?> type, String propertyName, IntegerConstraint ann) { + this(type, propertyName, + ann.min(), ann.max(), ann.allowed(), ann.disallowed()); + } + + /** + * @param type type of object that contains the constrained property + * @param propertyName name of property with constraint + * @param min minimum allowed value + * @param max maximum allowed value + * @param allowed optional set of allowed values + * @param disallowed optional set of disallowed values + */ + public Constraint(Class<?> type, String propertyName, + long min, long max, long[] allowed, long[] disallowed) { + mPropertyName = propertyName; + mMinValue = min; + mMaxValue = max; + if (mMaxValue < mMinValue) { + throw new MalformedTypeException + (type, "Illegal range for integer constraint on property \"" + + propertyName + "\": " + rangeString()); + } + + if (disallowed == null || disallowed.length == 0) { + disallowed = null; + } else { + disallowed = disallowed.clone(); + Arrays.sort(disallowed); + } + + if (allowed == null || allowed.length == 0) { + allowed = null; + } else { + allowed = allowed.clone(); + Arrays.sort(allowed); + for (long value : allowed) { + if (value < mMinValue || value > mMaxValue || + (disallowed != null && Arrays.binarySearch(disallowed, value) >= 0)) { + throw new MalformedTypeException + (type, "Allowed value contradiction for integer constraint " + + "on property \"" + propertyName + "\": " + value); + } + } + + // No need to have a set of disallowed values. + disallowed = null; + } + + mDisallowed = disallowed; + mAllowed = allowed; + } + + public void constrain(long propertyValue) { + if (propertyValue < mMinValue || propertyValue > mMaxValue) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" must be in range " + + rangeString() + ": " + propertyValue); + } + if (mDisallowed != null && Arrays.binarySearch(mDisallowed, propertyValue) >= 0) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is disallowed: " + propertyValue); + } + if (mAllowed != null && Arrays.binarySearch(mAllowed, propertyValue) < 0) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is not allowed: " + propertyValue); + } + } + + public void constrain(double propertyValue) { + if (propertyValue < mMinValue || propertyValue > mMaxValue) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" must be in range " + + rangeString() + ": " + propertyValue); + } + if (mDisallowed != null) { + long longValue = (long) propertyValue; + if (longValue == propertyValue && + Arrays.binarySearch(mDisallowed, longValue) >= 0) { + + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is disallowed: " + propertyValue); + } + } + if (mAllowed != null) { + long longValue = (long) propertyValue; + if (longValue != propertyValue || + Arrays.binarySearch(mAllowed, longValue) < 0) { + + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is not allowed: " + propertyValue); + } + } + } + + public void constrain(CharSequence propertyValue) { + if (propertyValue != null) { + try { + constrain(Long.parseLong(propertyValue.toString())); + } catch (NumberFormatException e) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is not an integer: " + + propertyValue); + } + } + } + + public void constrain(char propertyValue) { + if ('0' <= propertyValue && propertyValue <= '9') { + constrain((long) (propertyValue - '0')); + } else { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is not an integer: " + propertyValue); + } + } + + public void constrain(char[] propertyValue) { + if (propertyValue != null) { + constrain(new String(propertyValue)); + } + } + + private String rangeString() { + StringBuilder b = new StringBuilder(); + b.append('('); + if (mMinValue > Long.MIN_VALUE) { + b.append(mMinValue); + } + b.append(".."); + if (mMaxValue < Long.MAX_VALUE) { + b.append(mMaxValue); + } + b.append(')'); + return b.toString(); + } + } +} diff --git a/src/main/java/com/amazon/carbonado/constraint/LengthConstraint.java b/src/main/java/com/amazon/carbonado/constraint/LengthConstraint.java new file mode 100644 index 0000000..7e17a1f --- /dev/null +++ b/src/main/java/com/amazon/carbonado/constraint/LengthConstraint.java @@ -0,0 +1,173 @@ +/* + * Copyright 2006 Amazon Technologies, Inc. or its affiliates. + * Amazon, Amazon.com and Carbonado are trademarks or registered trademarks + * of Amazon Technologies, Inc. or its affiliates. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.amazon.carbonado.constraint; + +import java.lang.annotation.*; + +import com.amazon.carbonado.MalformedTypeException; + +/** + * Limits the value of a property to lie within a specific length range. The + * property value may be a String, CharSequence, or any kind of array. If the + * set property size is outside the range, an IllegalArgumentException is + * thrown. + * + * <p>Example:<pre> + * public interface UserInfo extends Storable { + * int getFirstName(); + * + * <b>@LengthConstraint(min=1, max=50)</b> + * void setFirstName(String name); + * + * ... + * } + * </pre> + * + * @author Brian S O'Neill + */ +@Documented +@Retention(RetentionPolicy.RUNTIME) +@Target(ElementType.METHOD) +@ConstraintDefinition +public @interface LengthConstraint { + /** + * Specify minimum allowed size for property. Default is zero. + */ + int min() default 0; + + /** + * Specify maximum allowed value for property. Default is unlimited. + */ + int max() default Integer.MAX_VALUE; + + /** + * Constraint implementation for {@link LengthConstraint}. + */ + public static class Constraint { + private final String mPropertyName; + private final int mMinLength; + private final int mMaxLength; + + /** + * @param type type of object that contains the constrained property + * @param propertyName name of property with constraint + * @param ann specific annotation that binds to this constraint class + */ + public Constraint(Class<?> type, String propertyName, LengthConstraint ann) { + this(type, propertyName, ann.min(), ann.max()); + } + + /** + * @param type type of object that contains the constrained property + * @param propertyName name of property with constraint + * @param min minimum allowed size + * @param max maximum allowed size + */ + public Constraint(Class<?> type, String propertyName, int min, int max) { + mPropertyName = propertyName; + mMinLength = min; + mMaxLength = max; + if (mMinLength < 0 || mMaxLength < mMinLength) { + throw new MalformedTypeException + (type, "Illegal length constraint for property \"" + propertyName + + "\": " + rangeString()); + } + } + + public void constrain(CharSequence str) { + if (str != null) { + constrainLength(str.length()); + } + } + + public void constrain(boolean[] array) { + if (array != null) { + constrainLength(array.length); + } + } + + public void constrain(byte[] array) { + if (array != null) { + constrainLength(array.length); + } + } + + public void constrain(short[] array) { + if (array != null) { + constrainLength(array.length); + } + } + + public void constrain(char[] array) { + if (array != null) { + constrainLength(array.length); + } + } + + public void constrain(int[] array) { + if (array != null) { + constrainLength(array.length); + } + } + + public void constrain(long[] array) { + if (array != null) { + constrainLength(array.length); + } + } + + public void constrain(float[] array) { + if (array != null) { + constrainLength(array.length); + } + } + + public void constrain(double[] array) { + if (array != null) { + constrainLength(array.length); + } + } + + public void constrain(Object[] array) { + if (array != null) { + constrainLength(array.length); + } + } + + private void constrainLength(int length) { + if (length < mMinLength || length > mMaxLength) { + throw new IllegalArgumentException + ("Value length for \"" + mPropertyName + "\" must be in range " + + rangeString() + ": " + length); + } + } + + private String rangeString() { + StringBuilder b = new StringBuilder(); + b.append('('); + b.append(mMinLength); + b.append(".."); + if (mMaxLength < Integer.MAX_VALUE) { + b.append(mMaxLength); + } + b.append(')'); + return b.toString(); + } + } +} diff --git a/src/main/java/com/amazon/carbonado/constraint/TextConstraint.java b/src/main/java/com/amazon/carbonado/constraint/TextConstraint.java new file mode 100644 index 0000000..08574e8 --- /dev/null +++ b/src/main/java/com/amazon/carbonado/constraint/TextConstraint.java @@ -0,0 +1,152 @@ +/* + * Copyright 2006 Amazon Technologies, Inc. or its affiliates. + * Amazon, Amazon.com and Carbonado are trademarks or registered trademarks + * of Amazon Technologies, Inc. or its affiliates. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.amazon.carbonado.constraint; + +import java.lang.annotation.*; +import java.util.Arrays; + +import com.amazon.carbonado.MalformedTypeException; + +/** + * Limits the value of a property to be a member of a specific set. The + * property value may be a String, CharSequence, char, Character, or character + * array. If the property value is outside the set, an IllegalArgumentException + * is thrown. + * + * <p>Example:<pre> + * public interface UserInfo extends Storable { + * char isActive(); + * + * <b>@TextConstraint(allowed={"Y", "N"})</b> + * void setActive(char value); + * + * ... + * } + * </pre> + * + * @author Brian S O'Neill + * @see IntegerConstraint + * @see FloatConstraint + */ +@Documented +@Retention(RetentionPolicy.RUNTIME) +@Target(ElementType.METHOD) +@ConstraintDefinition +public @interface TextConstraint { + /** + * Specific allowed values for property. Default is unlimited. + */ + String[] allowed() default {}; + + /** + * Specific disallowed values for property. Default is none. + */ + String[] disallowed() default {}; + + /** + * Constraint implementation for {@link TextConstraint}. + */ + public static class Constraint { + private final String mPropertyName; + + /** Allowed values, sorted for binary search. */ + private final String[] mAllowed; + + /** Disallowed values, sorted for binary search. */ + private final String[] mDisallowed; + + /** + * @param type type of object that contains the constrained property + * @param propertyName name of property with constraint + * @param ann specific annotation that binds to this constraint class + */ + public Constraint(Class<?> type, String propertyName, TextConstraint ann) { + this(type, propertyName, ann.allowed(), ann.disallowed()); + } + + /** + * @param type type of object that contains the constrained property + * @param propertyName name of property with constraint + * @param allowed optional set of allowed values + * @param disallowed optional set of disallowed values + */ + public Constraint(Class<?> type, String propertyName, + String[] allowed, String[] disallowed) { + mPropertyName = propertyName; + + if (disallowed == null || disallowed.length == 0) { + disallowed = null; + } else { + disallowed = disallowed.clone(); + Arrays.sort(disallowed); + } + + if (allowed == null || allowed.length == 0) { + allowed = null; + } else { + allowed = allowed.clone(); + Arrays.sort(allowed); + if (disallowed != null) { + for (String value : allowed) { + if (Arrays.binarySearch(disallowed, value) >= 0) { + throw new MalformedTypeException + (type, "Allowed value contradiction for text constraint " + + "on property \"" + propertyName + "\": " + value); + } + } + + // No need to have a set of disallowed values. + disallowed = null; + } + } + + mDisallowed = disallowed; + mAllowed = allowed; + } + + public void constrain(CharSequence propertyValue) { + constrain(propertyValue.toString()); + } + + public void constrain(String propertyValue) { + if (propertyValue == null) { + return; + } + if (mDisallowed != null && Arrays.binarySearch(mDisallowed, propertyValue) >= 0) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is disallowed: " + propertyValue); + } + if (mAllowed != null && Arrays.binarySearch(mAllowed, propertyValue) < 0) { + throw new IllegalArgumentException + ("Value for \"" + mPropertyName + "\" is not allowed: " + propertyValue); + } + } + + public void constrain(char propertyValue) { + constrain(Character.toString(propertyValue)); + } + + public void constrain(char[] propertyValue) { + if (propertyValue != null) { + constrain(new String(propertyValue)); + } + } + } +} + diff --git a/src/main/java/com/amazon/carbonado/constraint/package-info.java b/src/main/java/com/amazon/carbonado/constraint/package-info.java new file mode 100644 index 0000000..517c27e --- /dev/null +++ b/src/main/java/com/amazon/carbonado/constraint/package-info.java @@ -0,0 +1,24 @@ +/* + * Copyright 2006 Amazon Technologies, Inc. or its affiliates. + * Amazon, Amazon.com and Carbonado are trademarks or registered trademarks + * of Amazon Technologies, Inc. or its affiliates. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +/** + * Contains annotations and implementations for supporting property constraints. + * + * @see com.amazon.carbonado.constraint.ConstraintDefinition + */ +package com.amazon.carbonado.constraint; -- cgit v1.2.3