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

ezvcard.io.xml.XCardDocument Maven / Gradle / Ivy

Go to download

A library that reads and writes vCards, supporting all versions of the vCard standard (2.1, 3.0, and 4.0) as well as xCard (XML-encoded vCards), hCard (HTML-encoded vCards), and jCard (JSON-encoded vCards).

There is a newer version: 0.9.6
Show newest version
package ezvcard.io.xml;

import static ezvcard.io.xml.XCardQNames.GROUP;
import static ezvcard.io.xml.XCardQNames.PARAMETERS;
import static ezvcard.io.xml.XCardQNames.VCARD;
import static ezvcard.io.xml.XCardQNames.VCARDS;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.Reader;
import java.io.StringWriter;
import java.io.UncheckedIOException;
import java.io.Writer;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import javax.xml.namespace.QName;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerConfigurationException;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.TransformerFactoryConfigurationError;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;
import javax.xml.xpath.XPath;
import javax.xml.xpath.XPathConstants;
import javax.xml.xpath.XPathExpressionException;
import javax.xml.xpath.XPathFactory;

import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.xml.sax.SAXException;

import ezvcard.VCard;
import ezvcard.VCardDataType;
import ezvcard.VCardVersion;
import ezvcard.io.CannotParseException;
import ezvcard.io.EmbeddedVCardException;
import ezvcard.io.ParseWarning;
import ezvcard.io.SkipMeException;
import ezvcard.io.StreamReader;
import ezvcard.io.StreamWriter;
import ezvcard.io.scribe.VCardPropertyScribe;
import ezvcard.parameter.VCardParameters;
import ezvcard.property.VCardProperty;
import ezvcard.property.Xml;
import ezvcard.util.ListMultimap;
import ezvcard.util.XmlUtils;

/*
 Copyright (c) 2012-2023, Michael Angstadt
 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. 

 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 OWNER 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.

 The views and conclusions contained in the software and documentation are those
 of the authors and should not be interpreted as representing official policies, 
 either expressed or implied, of the FreeBSD Project.
 */

//@formatter:off
/**
* 

* Represents an XML document that contains vCard objects ("xCard" standard). * This class can be used to read and write xCard documents. *

*

* Examples: *

* *
* String xml =
* "<vcards xmlns=\"urn:ietf:params:xml:ns:vcard-4.0\">" +
*   "<vcard>" +
*     "<fn>" +
*       "<text>John Doe</text>" +
*     "</fn>" +
*     "<n>" +
*       "<surname>Doe</surname>" +
*        "<given>Johnathan</given>" +
*        "<additional>Jonny</additional>" +
*        "<additional>John</additional>" +
*        "<prefix>Mr.</prefix>" +
*        "<suffix />" +
*      "</n>" +
*    "</vcard>" +
* "</vcards>";
*     
* //parsing an existing xCard document
* XCardDocument xcard = new XCardDocument(xml);
* List<VCard> vcards = xcard.getVCards();
* 
* //creating an empty xCard document
* XCardDocument xcard = new XCardDocument();
* 
* //VCard objects can be added at any time
* VCard vcard = ...
* xcard.addVCard(vcard);
* 
* //retrieving the raw XML DOM
* Document document = xcard.getDocument();
* 
* //call one of the "write()" methods to output the xCard document
* File file = new File("johndoe.xml");
* xcard.write(file);
* 
* @author Michael Angstadt * @see RFC 6351 */ //@formatter:on public class XCardDocument { private final VCardVersion version4 = VCardVersion.V4_0; //xCard only supports 4.0 private final Document document; private Element vcardsRootElement; /** * Creates an empty xCard document. */ public XCardDocument() { document = XmlUtils.createDocument(); vcardsRootElement = document.createElementNS(VCARDS.getNamespaceURI(), VCARDS.getLocalPart()); document.appendChild(vcardsRootElement); } /** * Parses an xCard document from a string. * @param xml the XML string to read the vCards from * @throws SAXException if there's a problem parsing the XML */ public XCardDocument(String xml) throws SAXException { this(XmlUtils.toDocument(xml)); } /** * Parses an xCard document from an input stream. * @param in the input stream to read the vCards from * @throws IOException if there's a problem reading from the input stream * @throws SAXException if there's a problem parsing the XML */ public XCardDocument(InputStream in) throws SAXException, IOException { this(XmlUtils.toDocument(in)); } /** * Parses an xCard document from a file. * @param file the file to read the vCards from * @throws IOException if there's a problem reading from the file * @throws SAXException if there's a problem parsing the XML */ public XCardDocument(Path file) throws SAXException, IOException { this(XmlUtils.toDocument(file)); } /** *

* Parses an xCard document from a reader. *

*

* Note that use of this constructor is discouraged. It ignores the * character encoding that is defined within the XML document itself, and * should only be used if the encoding is undefined or if the encoding needs * to be ignored for whatever reason. The * {@link #XCardDocument(InputStream)} constructor is preferred, since it * takes the XML document's character encoding into account when parsing. *

* @param reader the reader to read the vCards from * @throws IOException if there's a problem reading from the reader * @throws SAXException if there's a problem parsing the XML */ public XCardDocument(Reader reader) throws SAXException, IOException { this(XmlUtils.toDocument(reader)); } /** * Wraps an existing XML DOM object. * @param document the XML DOM that contains the xCard document */ public XCardDocument(Document document) { this.document = document; XCardNamespaceContext nsContext = new XCardNamespaceContext(version4, "v"); XPath xpath = XPathFactory.newInstance().newXPath(); xpath.setNamespaceContext(nsContext); try { //find the element vcardsRootElement = (Element) xpath.evaluate("//" + nsContext.getPrefix() + ":" + VCARDS.getLocalPart(), document, XPathConstants.NODE); } catch (XPathExpressionException e) { //should never thrown because the xpath expression is hard coded throw new RuntimeException(e); } } /** * Creates a {@link StreamReader} object that reads vCards from this XML * document. * @return the reader */ public StreamReader reader() { return new XCardDocumentStreamReader(); } /** * Creates a {@link StreamWriter} object that adds vCards to this XML * document. * @return the writer */ public XCardDocumentStreamWriter writer() { return new XCardDocumentStreamWriter(); } /** * Gets the wrapped XML document. * @return the XML document */ public Document getDocument() { return document; } /** * Parses all of the vCards from this XML document. Modifications made to * these {@link VCard} objects will NOT be applied to the XML document. * @return the parsed vCards */ public List getVCards() { try { return reader().readAll(); } catch (IOException e) { //not thrown because we're reading from a DOM throw new UncheckedIOException(e); } } /** * Adds a vCard to the XML document. * @param vcard the vCard to add * @throws IllegalArgumentException if no scribe has been registered for the * property (only applies to custom property classes) */ public void addVCard(VCard vcard) { writer().write(vcard); } /** * Writes the XML document to a string. * @return the XML string */ public String write() { return write((Integer) null); } /** * Writes the XML document to a string. * @param indent the number of indent spaces to use for pretty-printing or * null to disable pretty-printing (disabled by default) * @return the XML string */ public String write(Integer indent) { return write(indent, null); } /** * Writes the XML document to a string. * @param indent the number of indent spaces to use for pretty-printing or * null to disable pretty-printing (disabled by default) * @param xmlVersion the XML version to use (defaults to "1.0") (Note: Many * JDKs only support 1.0 natively. For XML 1.1 support, add a JAXP library * like xalan to your project) * @return the XML string */ public String write(Integer indent, String xmlVersion) { return write(new XCardOutputProperties(indent, xmlVersion)); } /** * Writes the XML document to a string. * @param outputProperties properties to assign to the JAXP transformer (see * {@link Transformer#setOutputProperty}) * @return the XML string */ public String write(Map outputProperties) { StringWriter sw = new StringWriter(); try { write(sw, outputProperties); } catch (TransformerException e) { //should not be thrown because we're writing to a string throw new RuntimeException(e); } return sw.toString(); } /** * Writes the XML document to an output stream. * @param out the output stream (UTF-8 encoding will be used) * @throws TransformerException if there's a problem writing to the output * stream */ public void write(OutputStream out) throws TransformerException { write(out, (Integer) null); } /** * Writes the XML document to an output stream. * @param out the output stream (UTF-8 encoding will be used) * @param indent the number of indent spaces to use for pretty-printing or * null to disable pretty-printing (disabled by default) * @throws TransformerException if there's a problem writing to the output * stream */ public void write(OutputStream out, Integer indent) throws TransformerException { write(out, indent, null); } /** * Writes the XML document to an output stream. * @param out the output stream (UTF-8 encoding will be used) * @param indent the number of indent spaces to use for pretty-printing or * null to disable pretty-printing (disabled by default) * @param xmlVersion the XML version to use (defaults to "1.0") (Note: Many * JDKs only support 1.0 natively. For XML 1.1 support, add a JAXP library * like xalan to your project) * @throws TransformerException if there's a problem writing to the output * stream */ public void write(OutputStream out, Integer indent, String xmlVersion) throws TransformerException { write(out, new XCardOutputProperties(indent, xmlVersion)); } /** * Writes the XML document to an output stream. * @param out the output stream (UTF-8 encoding will be used) * @param outputProperties properties to assign to the JAXP transformer (see * {@link Transformer#setOutputProperty}) * @throws TransformerException if there's a problem writing to the output * stream */ public void write(OutputStream out, Map outputProperties) throws TransformerException { write(new OutputStreamWriter(out, StandardCharsets.UTF_8), outputProperties); } /** * Writes the XML document to a file. * @param file the file to write to (UTF-8 encoding will be used) * @throws TransformerException if there's a problem writing to the file * @throws IOException if there's a problem writing to the file */ public void write(Path file) throws TransformerException, IOException { write(file, (Integer) null); } /** * Writes the XML document to a file. * @param file the file to write to (UTF-8 encoding will be used) * @param indent the number of indent spaces to use for pretty-printing or * null to disable pretty-printing (disabled by default) * @throws TransformerException if there's a problem writing to the file * @throws IOException if there's a problem writing to the file */ public void write(Path file, Integer indent) throws TransformerException, IOException { write(file, indent, null); } /** * Writes the XML document to a file. * @param file the file to write to (UTF-8 encoding will be used) * @param indent the number of indent spaces to use for pretty-printing or * null to disable pretty-printing (disabled by default) * @param xmlVersion the XML version to use (defaults to "1.0") (Note: Many * JDKs only support 1.0 natively. For XML 1.1 support, add a JAXP library * like xalan to your project) * @throws TransformerException if there's a problem writing to the file * @throws IOException if there's a problem writing to the file */ public void write(Path file, Integer indent, String xmlVersion) throws TransformerException, IOException { write(file, new XCardOutputProperties(indent, xmlVersion)); } /** * Writes the XML document to a file. * @param file the file to write to (UTF-8 encoding will be used) * @param outputProperties properties to assign to the JAXP transformer (see * {@link Transformer#setOutputProperty}) * @throws TransformerException if there's a problem writing to the file * @throws IOException if there's a problem writing to the file */ public void write(Path file, Map outputProperties) throws TransformerException, IOException { try (Writer writer = Files.newBufferedWriter(file, StandardCharsets.UTF_8)) { write(writer, outputProperties); } } /** * Writes the XML document to a writer. * @param writer the writer * @throws TransformerException if there's a problem writing to the writer */ public void write(Writer writer) throws TransformerException { write(writer, (Integer) null); } /** * Writes the XML document to a writer. * @param writer the writer * @param indent the number of indent spaces to use for pretty-printing or * null to disable pretty-printing (disabled by default) * @throws TransformerException if there's a problem writing to the writer */ public void write(Writer writer, Integer indent) throws TransformerException { write(writer, indent, null); } /** * Writes the XML document to a writer. * @param writer the writer * @param indent the number of indent spaces to use for pretty-printing or * null to disable pretty-printing (disabled by default) * @param xmlVersion the XML version to use (defaults to "1.0") (Note: Many * JDKs only support 1.0 natively. For XML 1.1 support, add a JAXP library * like xalan to your project) * @throws TransformerException if there's a problem writing to the writer */ public void write(Writer writer, Integer indent, String xmlVersion) throws TransformerException { write(writer, new XCardOutputProperties(indent, xmlVersion)); } /** * Writes the XML document to a writer. * @param writer the writer * @param outputProperties properties to assign to the JAXP transformer (see * {@link Transformer#setOutputProperty}) * @throws TransformerException if there's a problem writing to the writer */ public void write(Writer writer, Map outputProperties) throws TransformerException { Transformer transformer; try { transformer = TransformerFactory.newInstance().newTransformer(); } catch (TransformerConfigurationException e) { //should never be thrown because we're not doing anything fancy with the configuration throw new RuntimeException(e); } catch (TransformerFactoryConfigurationError e) { //should never be thrown because we're not doing anything fancy with the configuration throw new RuntimeException(e); } /* * Using Transformer#setOutputProperties(Properties) doesn't work for * some reason for setting the number of indentation spaces. */ for (Map.Entry entry : outputProperties.entrySet()) { String key = entry.getKey(); String value = entry.getValue(); transformer.setOutputProperty(key, value); } DOMSource source = new DOMSource(document); StreamResult result = new StreamResult(writer); transformer.transform(source, result); } private class XCardDocumentStreamReader extends StreamReader { private final Iterator vcardElements; { List list = (vcardsRootElement == null) ? Collections. emptyList() : getChildElements(vcardsRootElement, VCARD); vcardElements = list.iterator(); } private VCard vcard; @Override public VCard readNext() { try { return super.readNext(); } catch (IOException e) { //will not be thrown throw new UncheckedIOException(e); } } @Override protected VCard _readNext() throws IOException { if (!vcardElements.hasNext()) { return null; } vcard = new VCard(); vcard.setVersion(version4); context.setVersion(version4); parseVCardElement(vcardElements.next()); return vcard; } public void close() { //empty } private void parseVCardElement(Element vcardElement) { List children = XmlUtils.toElementList(vcardElement.getChildNodes()); for (Element child : children) { if (XmlUtils.hasQName(child, GROUP)) { String group = child.getAttribute("name"); if (group.isEmpty()) { group = null; } List grandChildren = XmlUtils.toElementList(child.getChildNodes()); for (Element grandChild : grandChildren) { parseAndAddElement(grandChild, group); } continue; } parseAndAddElement(child, null); } } /** * Parses a property element from the XML document and adds the property * to the vCard. * @param element the element to parse * @param group the group name or null if the property does not belong * to a group */ private void parseAndAddElement(Element element, String group) { VCardParameters parameters = parseParameters(element); VCardProperty property; String propertyName = element.getLocalName(); String ns = element.getNamespaceURI(); QName qname = new QName(ns, propertyName); VCardPropertyScribe scribe = index.getPropertyScribe(qname); context.getWarnings().clear(); context.setPropertyName(propertyName); try { property = scribe.parseXml(element, parameters, context); property.setGroup(group); warnings.addAll(context.getWarnings()); } catch (SkipMeException e) { //@formatter:off warnings.add(new ParseWarning.Builder(context) .message(22, e.getMessage()) .build() ); //@formatter:on return; } catch (CannotParseException e) { //@formatter:off warnings.add(new ParseWarning.Builder(context) .message(e) .build() ); //@formatter:on scribe = index.getPropertyScribe(Xml.class); property = scribe.parseXml(element, parameters, context); property.setGroup(group); } catch (EmbeddedVCardException e) { //@formatter:off warnings.add(new ParseWarning.Builder(context) .message(34) .build() ); //@formatter:on return; } vcard.addProperty(property); } /** * Parses the property parameters. * @param element the property's XML element * @return the parsed parameters */ private VCardParameters parseParameters(Element element) { VCardParameters parameters = new VCardParameters(); List roots = XmlUtils.toElementList(element.getElementsByTagNameNS(PARAMETERS.getNamespaceURI(), PARAMETERS.getLocalPart())); for (Element root : roots) { // foreach "" element (there should only be 1 though) List parameterElements = XmlUtils.toElementList(root.getChildNodes()); for (Element parameterElement : parameterElements) { String name = parameterElement.getLocalName().toUpperCase(); List valueElements = XmlUtils.toElementList(parameterElement.getChildNodes()); for (Element valueElement : valueElements) { String value = valueElement.getTextContent(); parameters.put(name, value); } } } return parameters; } private List getChildElements(Element parent, QName qname) { List elements = new ArrayList<>(); for (Element child : XmlUtils.toElementList(parent.getChildNodes())) { if (XmlUtils.hasQName(child, qname)) { elements.add(child); } } return elements; } } public class XCardDocumentStreamWriter extends XCardWriterBase { @Override public void write(VCard vcard) { try { super.write(vcard); } catch (IOException ignore) { //won't be thrown because we're writing to a DOM } } @Override protected void _write(VCard vcard, List properties) throws IOException { //group properties by group name (null = no group name) ListMultimap propertiesByGroup = new ListMultimap<>(); for (VCardProperty property : properties) { propertiesByGroup.put(property.getGroup(), property); } //marshal each property object Element vcardElement = createElement(VCARD); for (Map.Entry> entry : propertiesByGroup) { String groupName = entry.getKey(); Element parent; if (groupName != null) { Element groupElement = createElement(GROUP); groupElement.setAttribute("name", groupName); vcardElement.appendChild(groupElement); parent = groupElement; } else { parent = vcardElement; } for (VCardProperty property : entry.getValue()) { try { Element propertyElement = marshalProperty(property, vcard); parent.appendChild(propertyElement); } catch (SkipMeException e) { //skip property } catch (EmbeddedVCardException e) { //skip property } } } if (vcardsRootElement == null) { vcardsRootElement = createElement(VCARDS); Element documentRoot = document.getDocumentElement(); if (documentRoot == null) { document.appendChild(vcardsRootElement); } else { documentRoot.appendChild(vcardsRootElement); } } vcardsRootElement.appendChild(vcardElement); } public void close() { //empty } /** * Marshals a type object to an XML element. * @param property the property to marshal * @param vcard the vcard the type belongs to * @return the XML element */ @SuppressWarnings({ "rawtypes", "unchecked" }) private Element marshalProperty(VCardProperty property, VCard vcard) { VCardPropertyScribe scribe = index.getPropertyScribe(property); Element propertyElement; if (property instanceof Xml) { Xml xml = (Xml) property; Document propertyDocument = xml.getValue(); if (propertyDocument == null) { throw new SkipMeException(); } propertyElement = propertyDocument.getDocumentElement(); propertyElement = (Element) document.importNode(propertyElement, true); } else { QName qname = scribe.getQName(); propertyElement = createElement(qname); scribe.writeXml(property, propertyElement); } //marshal the parameters VCardParameters parameters = scribe.prepareParameters(property, targetVersion, vcard); removeUnsupportedParameters(parameters); if (!parameters.isEmpty()) { Element parametersElement = marshalParameters(parameters); Node firstChild = propertyElement.getFirstChild(); propertyElement.insertBefore(parametersElement, firstChild); } return propertyElement; } private Element marshalParameters(VCardParameters parameters) { Element parametersElement = createElement(PARAMETERS); for (Map.Entry> parameter : parameters) { String parameterName = parameter.getKey().toLowerCase(); Element parameterElement = createElement(parameterName); for (String parameterValue : parameter.getValue()) { VCardDataType dataType = parameterDataTypes.get(parameterName); String dataTypeElementName = (dataType == null) ? "unknown" : dataType.getName().toLowerCase(); Element dataTypeElement = createElement(dataTypeElementName); dataTypeElement.setTextContent(parameterValue); parameterElement.appendChild(dataTypeElement); } parametersElement.appendChild(parameterElement); } return parametersElement; } /** * Creates a new XML element under the vCard namespace. * @param name the name of the XML element * @return the new XML element */ private Element createElement(String name) { return createElement(name, targetVersion.getXmlNamespace()); } /** * Creates a new XML element. * @param name the name of the XML element * @param ns the namespace of the XML element * @return the new XML element */ private Element createElement(String name, String ns) { return document.createElementNS(ns, name); } /** * Creates a new XML element. * @param qname the element name * @return the new XML element */ private Element createElement(QName qname) { return createElement(qname.getLocalPart(), qname.getNamespaceURI()); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy