All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.jcabi.xml.ListWrapper Maven / Gradle / Ivy

There is a newer version: 0.30.1
Show newest version
/**
 * Copyright (c) 2012-2017, jcabi.com
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met: 1) Redistributions of source code must retain the above
 * copyright notice, this list of conditions and the following
 * disclaimer. 2) Redistributions in binary form must reproduce the above
 * copyright notice, this list of conditions and the following
 * disclaimer in the documentation and/or other materials provided
 * with the distribution. 3) Neither the name of the jcabi.com nor
 * the names of its contributors may be used to endorse or promote
 * products derived from this software without specific prior written
 * permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT
 * NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
 * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
 * THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
 * OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package com.jcabi.xml;

import com.jcabi.log.Logger;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;
import lombok.EqualsAndHashCode;
import org.w3c.dom.Node;

/**
 * Wrapper of {@link List}.
 *
 * 

This wrapper is our internal implementation of a {@link List}. The only * purpose of this wrapper is to throw our own custom exception when the client * is trying to access an element that is absent in the list. Such a custom * exception ({@link ListWrapper.NodeNotFoundException}) * includes detailed information about * the original document. Thus, such an incorrect list-access operation will * lead to an exception that contains all the details inside (not just a simple * error message). For example: * *

 String name = new XMLDocument("...")
 *   .xpath("/document/name/text()")
 *   .get(0)
* *

This snippet is trying to get a text value from XML element, and there * is no guarantee that such an element exists in the document. In order to give * a detailed report to the user of the problem (including a full content of * the XML document) we're returning {@link ListWrapper} from * {@link XMLDocument#xpath(String)}. The only method that we implement * for this purpose is {@link #get(int)}. * *

{@link ListWrapper} is an unmodifiable list, that's why * the majority of inherited method are not implemented and * thow runtime exceptions if being called. * *

The method {@link #get(int)} throws * {@link ListWrapper.NodeNotFoundException} * if such an element doesn't exist in the list. * *

The method {@link #subList(int, int)} * throws {@link ListWrapper.NodeNotFoundException} * when either * {@code start} or {@code end} is bigger than the size of the list. In all * other cases of illegal method call (start is less than zero, end is * less than zero, or start is bigger than end) a standard * {@link IndexOutOfBoundsException} is thrown (by the encapsulated * implementation of {@Link List}). * *

The class is immutable and thread-safe. * * @author Yegor Bugayenko ([email protected]) * @version $Id: 600a8cbfbd961e301001e7d11f1a43cdb70e3c1d $ * @since 0.1 * @param Time of items */ @EqualsAndHashCode(of = { "original", "dom", "xpath" }) @SuppressWarnings("PMD.TooManyMethods") final class ListWrapper implements List { /** * The original list. */ private final transient List original; /** * The XML where this list came from. */ private final transient Node dom; /** * XPath. */ private final transient String xpath; /** * Public ctor. * @param list Original list * @param node The XML * @param addr Address */ ListWrapper(final List list, final Node node, final String addr) { this.original = list; this.dom = node; this.xpath = addr; } @Override public String toString() { return this.original.toString(); } @Override public boolean add(final T element) { throw new UnsupportedOperationException("#add(T)"); } @Override public void add(final int index, final T element) { throw new UnsupportedOperationException("#add(int, T)"); } @Override public boolean addAll(final Collection elements) { throw new UnsupportedOperationException("#addAll(Collection)"); } @Override public boolean addAll(final int index, final Collection elms) { throw new UnsupportedOperationException("#add(int, Collection)"); } @Override public void clear() { throw new UnsupportedOperationException("#clear()"); } @Override public boolean contains(final Object element) { return this.original.contains(element); } @Override public boolean containsAll(final Collection elements) { return this.original.containsAll(elements); } @Override public T get(final int index) { if (index >= this.size()) { throw new ListWrapper.NodeNotFoundException( String.format( "Index (%d) is out of bounds (size=%d)", index, this.size() ), this.dom, this.xpath ); } return this.original.get(index); } @Override public int indexOf(final Object element) { return this.original.indexOf(element); } @Override public boolean isEmpty() { return this.original.isEmpty(); } @Override public Iterator iterator() { return this.original.iterator(); } @Override public int lastIndexOf(final Object element) { return this.original.lastIndexOf(element); } @Override public ListIterator listIterator() { return this.original.listIterator(); } @Override public ListIterator listIterator(final int index) { return this.original.listIterator(index); } @Override public T remove(final int index) { throw new UnsupportedOperationException("#remove(int)"); } @Override public boolean remove(final Object element) { throw new UnsupportedOperationException("#remove(Object)"); } @Override public boolean removeAll(final Collection elements) { throw new UnsupportedOperationException("#removeAll(Collection)"); } @Override public boolean retainAll(final Collection elements) { throw new UnsupportedOperationException("#retainAll(Collection)"); } @Override public T set(final int index, final T element) { throw new UnsupportedOperationException("#set(int, T)"); } @Override public int size() { return this.original.size(); } @Override public List subList(final int start, final int end) { if (start >= this.size()) { throw new ListWrapper.NodeNotFoundException( String.format( "Start of subList (%d) is out of bounds (size=%d)", start, this.size() ), this.dom, this.xpath ); } if (end >= this.size()) { throw new ListWrapper.NodeNotFoundException( String.format( "End of subList (%d) is out of bounds (size=%d)", end, this.size() ), this.dom, this.xpath ); } return this.original.subList(start, end); } @Override public Object[] toArray() { return this.original.toArray(); } @Override @SuppressWarnings("PMD.UseVarargs") public E[] toArray(final E[] array) { return this.original.toArray(array); } /** * Node not found in XmlDocument. */ private static final class NodeNotFoundException extends IndexOutOfBoundsException { /** * Serialization marker. */ private static final long serialVersionUID = 0x7526FA78EEDAC470L; /** * Public ctor. * @param message Error message * @param node The XML with error * @param query The query in XPath */ NodeNotFoundException(final String message, final Node node, final CharSequence query) { super( Logger.format( "XPath '%s' not found in '%[text]s': %s", ListWrapper.NodeNotFoundException.escapeUnicode(query), ListWrapper.NodeNotFoundException.escapeUnicode( new XMLDocument(node).toString() ), message ) ); } /** * Escape unicode characters. * @param input Input string * @return Escaped output */ private static String escapeUnicode(final CharSequence input) { final int length = input.length(); final StringBuilder output = new StringBuilder(length); for (int index = 0; index < length; index += 1) { final char character = input.charAt(index); // @checkstyle MagicNumber (1 line) if (character < 32 || character > 0x7f) { output.append(String.format("\\u%X", (int) character)); } else { output.append(character); } } return output.toString(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy