blob: b7ba09e5c40e2138aec98f07fb7508ebaed12dff [file]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.jasper.util;
import java.util.HashSet;
import java.util.Set;
import org.apache.jasper.compiler.Localizer;
import org.xml.sax.Attributes;
import org.xml.sax.helpers.AttributesImpl;
/**
* Wraps the default attributes implementation and ensures that each attribute has a unique qname as required by the JSP
* specification.
*/
public class UniqueAttributesImpl extends AttributesImpl {
private static final String IMPORT = "import";
private static final String PAGE_ENCODING = "pageEncoding";
private final boolean pageDirective;
private final Set<String> qNames = new HashSet<>();
/**
* Constructs a new UniqueAttributesImpl for non-page directives.
*/
public UniqueAttributesImpl() {
this.pageDirective = false;
}
/**
* Constructs a new UniqueAttributesImpl.
*
* @param pageDirective true if processing a page directive
*/
public UniqueAttributesImpl(boolean pageDirective) {
this.pageDirective = pageDirective;
}
@Override
public void clear() {
qNames.clear();
super.clear();
}
@Override
public void setAttributes(Attributes atts) {
for (int i = 0; i < atts.getLength(); i++) {
if (!qNames.add(atts.getQName(i))) {
handleDuplicate(atts.getQName(i), atts.getValue(i));
}
}
super.setAttributes(atts);
}
@Override
public void addAttribute(String uri, String localName, String qName, String type, String value) {
if (qNames.add(qName)) {
super.addAttribute(uri, localName, qName, type, value);
} else {
handleDuplicate(qName, value);
}
}
@Override
public void setAttribute(int index, String uri, String localName, String qName, String type, String value) {
qNames.remove(super.getQName(index));
if (qNames.add(qName)) {
super.setAttribute(index, uri, localName, qName, type, value);
} else {
handleDuplicate(qName, value);
}
}
@Override
public void removeAttribute(int index) {
qNames.remove(super.getQName(index));
super.removeAttribute(index);
}
@Override
public void setQName(int index, String qName) {
qNames.remove(super.getQName(index));
super.setQName(index, qName);
}
private void handleDuplicate(String qName, String value) {
if (pageDirective) {
if (IMPORT.equalsIgnoreCase(qName)) {
// Always merge imports
int i = super.getIndex(IMPORT);
String v = super.getValue(i);
super.setValue(i, v + "," + value);
return;
} else if (PAGE_ENCODING.equalsIgnoreCase(qName)) {
// Page encoding can only occur once per file so a second
// attribute - even one with a duplicate value - is an error
} else {
// Other attributes can be repeated if and only if the values
// are identical
String v = super.getValue(qName);
if (v.equals(value)) {
return;
}
}
}
// Ordinary tag attributes can't be repeated, even with identical values
throw new IllegalArgumentException(Localizer.getMessage("jsp.error.duplicateqname", qName));
}
}