BUG-8039: Enforce binary/string type length
[yangtools.git] / yang / yang-data-impl / src / main / java / org / opendaylight / yangtools / yang / data / impl / codec / StringStringCodec.java
index d7bbb6971913aac06087cbe87af03fa8ddfb1be6..57602161e3c17f29151559ead11b1ea6355c3373 100644 (file)
@@ -8,27 +8,65 @@
 package org.opendaylight.yangtools.yang.data.impl.codec;
 
 import com.google.common.base.Optional;
+import com.google.common.base.Preconditions;
+import com.google.common.collect.ImmutableRangeSet;
+import com.google.common.collect.Range;
+import com.google.common.collect.RangeSet;
+import com.google.common.collect.TreeRangeSet;
+import java.util.Collection;
+import java.util.Objects;
 import org.opendaylight.yangtools.yang.data.api.codec.StringCodec;
+import org.opendaylight.yangtools.yang.model.api.type.LengthConstraint;
 import org.opendaylight.yangtools.yang.model.api.type.StringTypeDefinition;
 
 class StringStringCodec extends TypeDefinitionAwareCodec<String, StringTypeDefinition> implements
         StringCodec<String> {
 
-    protected StringStringCodec(final Optional<StringTypeDefinition> typeDef) {
-        super(typeDef, String.class);
+    private final RangeSet<Integer> lengths;
+
+    StringStringCodec(final StringTypeDefinition typeDef) {
+        super(Optional.of(typeDef), String.class);
+
+        final Collection<LengthConstraint> constraints = typeDef.getLengthConstraints();
+        if (!constraints.isEmpty()) {
+            final RangeSet<Integer> tmp = TreeRangeSet.create();
+            for (LengthConstraint c : constraints) {
+                tmp.add(Range.closed(c.getMin().intValue(), c.getMax().intValue()));
+            }
+
+            lengths = ImmutableRangeSet.copyOf(tmp);
+        } else {
+            lengths = null;
+        }
     }
 
-    static TypeDefinitionAwareCodec<?,StringTypeDefinition> from(final StringTypeDefinition normalizedType) {
-        return new StringStringCodec(Optional.fromNullable(normalizedType));
+    static TypeDefinitionAwareCodec<?, StringTypeDefinition> from(final StringTypeDefinition normalizedType) {
+        if (normalizedType.getPatternConstraints().isEmpty()) {
+            return new StringStringCodec(normalizedType);
+        }
+
+        return new StringPatternCheckingCodec(normalizedType);
     }
 
     @Override
     public final String deserialize(final String stringRepresentation) {
-        return stringRepresentation == null ? "" : stringRepresentation;
+        if (stringRepresentation == null) {
+            // FIXME: These seems buggy, but someone may be using this behaviour
+            return "";
+        }
+        validate(stringRepresentation);
+        return stringRepresentation;
     }
 
     @Override
     public final String serialize(final String data) {
-        return data == null ? "" : data;
+        return Objects.toString(data, "");
+    }
+
+    void validate(final String s) {
+        if (lengths != null) {
+            Preconditions.checkArgument(lengths.contains(s.length()), "String '%s' does not match allowed lengths %s",
+                lengths);
+        }
     }
-}
\ No newline at end of file
+}