/* * 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.jackrabbit.webdav.observation; import org.apache.jackrabbit.webdav.xml.Namespace; import org.apache.jackrabbit.webdav.xml.DomUtil; import org.apache.jackrabbit.webdav.xml.XmlSerializable; import org.apache.jackrabbit.webdav.xml.ElementIterator; import org.w3c.dom.Element; import org.w3c.dom.Document; import java.util.Map; import java.util.HashMap; import java.util.List; import java.util.ArrayList; /** * <code>DefaultEventType</code> defines a simple EventType implementation that * only consists of a qualified event name consisting of namespace plus local * name. */ public class DefaultEventType implements EventType { private static final Map<String, EventType> eventTypes = new HashMap<String, EventType>(); private final String localName; private final Namespace namespace; /** * Avoid instantiation of <code>DefaultEventType</code>. Since the amount * of available (and registered) events is considered to be limited, the * static {@link #create(String, Namespace) method is defined. * * @param localName * @param namespace */ private DefaultEventType(String localName, Namespace namespace) { this.localName = localName; this.namespace = namespace; } /** * Factory method to create a new <code>EventType</code>. * * @param localName * @param namespace * @return */ public static EventType create(String localName, Namespace namespace) { if (localName == null || "".equals(localName)) { throw new IllegalArgumentException("null and '' are not valid local names of an event type."); } String key = DomUtil.getExpandedName(localName, namespace); if (eventTypes.containsKey(key)) { return eventTypes.get(key); } else { EventType type = new DefaultEventType(localName, namespace); eventTypes.put(key, type); return type; } } /** * Factory method to create an array of new <code>EventType</code> for the * specified localNames and the specified namespace. * * @param localNames * @param namespace * @return An array of event types. */ public static EventType[] create(String[] localNames, Namespace namespace) { EventType[] types = new EventType[localNames.length]; for (int i = 0; i < localNames.length; i++) { types[i] = create(localNames[i], namespace); } return types; } /** * Retrieves one or multiple <code>EventType</code>s from the 'eventtype' * Xml element. While a subscription may register multiple types (thus * the 'eventtype' contains multiple child elements), a single event may only * refer to one single type. * * @param eventType * @return */ public static EventType[] createFromXml(Element eventType) { if (!DomUtil.matches(eventType, ObservationConstants.XML_EVENTTYPE, ObservationConstants.NAMESPACE)) { throw new IllegalArgumentException("'eventtype' element expected which contains a least a single child element."); } List<EventType> etypes = new ArrayList<EventType>(); ElementIterator it = DomUtil.getChildren(eventType); while (it.hasNext()) { Element el = it.nextElement(); etypes.add(create(el.getLocalName(), DomUtil.getNamespace(el))); } return etypes.toArray(new EventType[etypes.size()]); } //----------------------------------------------------------< EventType >--- /** * @see EventType#getName() */ public String getName() { return localName; } /** * @see EventType#getNamespace() */ public Namespace getNamespace() { return namespace; } //----------------------------------------------------< XmlSerializable >--- /** * Returns a single empty Xml element where namespace and local name of this * event type define the elements name. * <pre> * EventType.create("someevent", Namespace.getNamespace("F", "http://www.foo.bar/eventtypes")); * * returns the following element upon call of toXml: * * <F:someevent xmlns:F="http://www.foo.bar/eventtypes" /> * </pre> * * @see XmlSerializable#toXml(Document) */ public Element toXml(Document document) { return DomUtil.createElement(document, localName, namespace); } }