import org.opendaylight.yangtools.yang.model.api.stmt.DescriptionStatement;
import org.opendaylight.yangtools.yang.model.api.stmt.ErrorAppTagStatement;
import org.opendaylight.yangtools.yang.model.api.stmt.ErrorMessageStatement;
+import org.opendaylight.yangtools.yang.model.api.stmt.ModifierStatement;
import org.opendaylight.yangtools.yang.model.api.stmt.PatternStatement;
import org.opendaylight.yangtools.yang.model.api.stmt.ReferenceStatement;
import org.opendaylight.yangtools.yang.model.api.type.PatternConstraint;
@Override
public PatternConstraint parseArgumentValue(final StmtContext<?, ?, ?> ctx, final String value) {
- final String pattern = "^" + Utils.fixUnicodeScriptPattern(value) + '$';
+ final String pattern = getJavaRegexFromXSD(value);
try {
Pattern.compile(pattern);
- } catch (PatternSyntaxException e) {
+ } catch (final PatternSyntaxException e) {
LOG.debug("Pattern \"{}\" failed to compile at {}", pattern, ctx.getStatementSourceReference(), e);
return null;
}
- return new PatternConstraintEffectiveImpl(pattern, Optional.absent(), Optional.absent());
+ return new PatternConstraintEffectiveImpl(pattern, value, Optional.absent(), Optional.absent());
+ }
+
+ static String getJavaRegexFromXSD(final String xsdRegex) {
+ return "^" + Utils.fixUnicodeScriptPattern(escapeChars(xsdRegex)) + '$';
+ }
+
+ /*
+ * As both '^' and '$' are special anchor characters in java regular
+ * expressions which are implicitly present in XSD regular expressions,
+ * we need to escape them in case they are not defined as part of
+ * character ranges i.e. inside regular square brackets.
+ */
+ private static String escapeChars(final String regex) {
+ final StringBuilder result = new StringBuilder(regex.length());
+ int bracket = 0;
+ boolean escape = false;
+ for (int i = 0; i < regex.length(); i++) {
+ final char ch = regex.charAt(i);
+ switch (ch) {
+ case '[':
+ if (!escape) {
+ bracket++;
+ }
+ escape = false;
+ result.append(ch);
+ break;
+ case ']':
+ if (!escape) {
+ bracket--;
+ }
+ escape = false;
+ result.append(ch);
+ break;
+ case '\\':
+ escape = !escape;
+ result.append(ch);
+ break;
+ case '^':
+ case '$':
+ if (bracket == 0) {
+ result.append('\\');
+ }
+ escape = false;
+ result.append(ch);
+ break;
+ default:
+ escape = false;
+ result.append(ch);
+ }
+ }
+ return result.toString();
}
@Override
}
@Override
- public void onFullDefinitionDeclared(final StmtContext.Mutable<PatternConstraint, PatternStatement,
- EffectiveStatement<PatternConstraint, PatternStatement>> stmt) {
- super.onFullDefinitionDeclared(stmt);
- SUBSTATEMENT_VALIDATOR.validate(stmt);
+ protected SubstatementValidator getSubstatementValidator() {
+ return SUBSTATEMENT_VALIDATOR;
}
}
return firstDeclared(DescriptionStatement.class);
}
+ @Override
+ public ModifierStatement getModifierStatement() {
+ return firstDeclared(ModifierStatement.class);
+ }
+
@Override
public ReferenceStatement getReference() {
return firstDeclared(ReferenceStatement.class);
public PatternConstraint getValue() {
return argument();
}
-
}