/* * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. * * Copyright (c) 1997-2010 Oracle and/or its affiliates. All rights reserved. * * The contents of this file are subject to the terms of either the GNU * General Public License Version 2 only ("GPL") or the Common Development * and Distribution License("CDDL") (collectively, the "License"). You * may not use this file except in compliance with the License. You can * obtain a copy of the License at * https://glassfish.java.net/public/CDDL+GPL_1_1.html * or packager/legal/LICENSE.txt. See the License for the specific * language governing permissions and limitations under the License. * * When distributing the software, include this License Header Notice in each * file and include the License file at packager/legal/LICENSE.txt. * * GPL Classpath Exception: * Oracle designates this particular file as subject to the "Classpath" * exception as provided by Oracle in the GPL Version 2 section of the License * file that accompanied this code. * * Modifications: * If applicable, add the following below the License Header, with the fields * enclosed by brackets [] replaced by your own identifying information: * "Portions Copyright [year] [name of copyright owner]" * * Contributor(s): * If you wish your version of this file to be governed by only the CDDL or * only the GPL Version 2, indicate your decision by adding "[Contributor] * elects to include this software in this distribution under the [CDDL or GPL * Version 2] license." If you don't indicate a single choice of license, a * recipient has the option to distribute your version of this file under * either the CDDL, the GPL Version 2 or to extend the choice of license to * its licensees as provided above. However, if you add GPL Version 2 code * and therefore, elected the GPL Version 2 license, then the option applies * only if the new code is made subject to such option by the copyright * holder. */ package carstore; import javax.el.ValueExpression; import javax.faces.application.FacesMessage; import javax.faces.component.StateHolder; import javax.faces.component.UIComponent; import javax.faces.component.UIOutput; import javax.faces.context.FacesContext; import javax.faces.validator.Validator; import javax.faces.validator.ValidatorException; import java.util.ArrayList; /** * <p><strong>FormatValidator</strong> is a Validator that checks * the validity of String representation of the value of the * associated component against a list of specified patterns.</p> * <ul> * <li>Call getValue() to retrieve the current value of the component. * If it is <code>null</code>, exit immediately. (If null values * should not be allowed, a RequiredValidator can be configured * to check for this case.)</li> * <li><code>formatPattern</code> is a <code>|</code> separated string * of allowed patterns. </li> * <li> This validator uses the following rules to match a value against a * pattern. * <li> if the matching pattern has a "A", then corresponding character * in input value should be a letter. * <li> if the matching pattern has a "9", then corresponding character * in input value should be a number. * <li> if the matching pattern has a "#", then corresponding character * in input value should be a number or a letter. * <li> Any other character must match literally. * </ul> </ul> * <p/> * Validators have to be Serializable, so you can't maintain a reference to * a java.sql.Connection or javax.sql.DataSource inside this class in case * you need to hook upto the database or some other back end resource. * One approach would be to use JNDI-based data source lookups or do * this verification in the business tier. */ public class FormatValidator implements Validator, StateHolder { // ----------------------------------------------------- Manifest Constants /** * <p>The message identifier of the Message to be created if * the validation fails. The message format string for this * message may optionally include a <code>{0}</code> placeholder, which * will be replaced by list of format patterns.</p> */ public static final String FORMAT_INVALID_MESSAGE_ID = "carstore.Format_Invalid"; private ArrayList<String> formatPatternsList; // // Constructors and Initializers // public FormatValidator() { super(); } // // General Methods // /** * <code>|</code> separated String of format patterns * that this validator must match against. */ private ValueExpression formatPatterns = null; /* <p>Return the format patterns that the validator supports. */ /** * @return the configured format patterns */ public ValueExpression getFormatPatterns() { return (this.formatPatterns); } /** * <p>Set the format patterns that the validator support..</p> * * @param formatPatterns <code>|</code> separated String of format patterns * that this validator must match against. */ public void setFormatPatterns(ValueExpression formatPatterns) { this.formatPatterns = formatPatterns; parseFormatPatterns(); } /** * Parses the <code>formatPatterns</code> into validPatterns * <code>ArrayList</code>. The delimiter must be "|". */ public void parseFormatPatterns() { String evalPatterns = (String) formatPatterns .getValue(FacesContext.getCurrentInstance().getELContext()); if (evalPatterns == null || evalPatterns.length() == 0) { return; } String[] patterns = evalPatterns.split("\\|"); if (formatPatternsList != null) { return; } else { formatPatternsList = new ArrayList<String>(patterns.length); } for (String pattern : patterns) { formatPatternsList.add(pattern); } } // // Methods from Validator // public void validate(FacesContext context, UIComponent component, Object toValidate) { boolean valid = false; String value; if ((context == null) || (component == null)) { throw new NullPointerException(); } if (!(component instanceof UIOutput)) { return; } if (null == formatPatternsList || null == toValidate) { return; } value = toValidate.toString(); // validate the value against the list of valid patterns. for (String aFormatPatternsList : formatPatternsList) { valid = isFormatValid((aFormatPatternsList), value); if (valid) { break; } } if (!valid) { FacesMessage errMsg = MessageFactory.getMessage(context, FORMAT_INVALID_MESSAGE_ID, formatPatterns); throw new ValidatorException(errMsg); } } /* Returns true if the value matches one of the valid patterns. */ /** * <p>Returns true if the value matches one of the valid patterns.</p> * @param pattern the 'control' pattern * @param value the value to match * @return true if the value matches the pattern */ protected boolean isFormatValid(String pattern, String value) { boolean valid = true; // if there is no pattern to match then value is valid if (pattern == null || pattern.length() == 0) { return true; } // if the value is null or a zero length string return false. if (value == null || value.length() == 0) { return false; } // if the length of the value is not equal to the length of the // pattern string then the value is not valid. if (value.length() != pattern.length()) { return false; } value = value.trim(); // rules for matching. // 1. if the matching pattern has a "A", then corresponding character // in the value should a letter. // 2. if the matching pattern has a "9", then corresponding character // in the value should a number // 3. if the matching pattern has a "#", then corresponding character // in the value should a number or a letter // 4.. any other character must match literally. char[] input = value.toCharArray(); char[] fmtpattern = pattern.toCharArray(); for (int i = 0; i < fmtpattern.length; ++i) { if (fmtpattern[i] == 'A') { if (!(Character.isLetter(input[i]))) { valid = false; } } else if (fmtpattern[i] == '9') { if (!(Character.isDigit(input[i]))) { valid = false; } } else if (fmtpattern[i] == '#') { if ((!(Character.isDigit(input[i]))) && (!(Character.isLetter(input[i])))) { valid = false; } } else { if (!(fmtpattern[i] == input[i])) { valid = false; } } } return valid; } public Object saveState(FacesContext context) { if (context == null) { throw new NullPointerException(); } Object values[] = new Object[2]; values[0] = formatPatterns; values[1] = formatPatternsList; return (values); } public void restoreState(FacesContext context, Object state) { if (context == null) { throw new NullPointerException(); } if (state == null) { return; } Object values[] = (Object[]) state; formatPatterns = (ValueExpression) values[0]; formatPatternsList = (ArrayList<String>) values[1]; } private boolean transientValue = false; public boolean isTransient() { return (this.transientValue); } public void setTransient(boolean transientValue) { this.transientValue = transientValue; } }