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

org.postgresql.jdbc.PgSQLXML Maven / Gradle / Ivy

/*
 * Copyright (c) 2004, PostgreSQL Global Development Group
 * See the LICENSE file in the project root for more information.
 */

package org.postgresql.jdbc;

import org.postgresql.core.BaseConnection;
import org.postgresql.util.GT;
import org.postgresql.util.PSQLException;
import org.postgresql.util.PSQLState;
import org.postgresql.xml.DefaultPGXmlFactoryFactory;
import org.postgresql.xml.PGXmlFactoryFactory;

import org.checkerframework.checker.nullness.qual.Nullable;
import org.xml.sax.InputSource;
import org.xml.sax.XMLReader;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.Reader;
import java.io.StringReader;
import java.io.StringWriter;
import java.io.Writer;
import java.sql.SQLException;
import java.sql.SQLXML;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLOutputFactory;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.XMLStreamReader;
import javax.xml.stream.XMLStreamWriter;
import javax.xml.transform.Result;
import javax.xml.transform.Source;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMResult;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.sax.SAXResult;
import javax.xml.transform.sax.SAXSource;
import javax.xml.transform.sax.SAXTransformerFactory;
import javax.xml.transform.sax.TransformerHandler;
import javax.xml.transform.stax.StAXResult;
import javax.xml.transform.stax.StAXSource;
import javax.xml.transform.stream.StreamResult;
import javax.xml.transform.stream.StreamSource;

public class PgSQLXML implements SQLXML {

  private final ResourceLock lock = new ResourceLock();
  private final BaseConnection conn;
  private @Nullable String data; // The actual data contained.
  private boolean initialized; // Has someone assigned the data for this object?
  private boolean active; // Is anyone in the process of loading data into us?
  private boolean freed;

  private @Nullable ByteArrayOutputStream byteArrayOutputStream;
  private @Nullable StringWriter stringWriter;
  private @Nullable DOMResult domResult;

  public PgSQLXML(BaseConnection conn) {
    this(conn, null, false);
  }

  public PgSQLXML(BaseConnection conn, @Nullable String data) {
    this(conn, data, true);
  }

  private PgSQLXML(BaseConnection conn, @Nullable String data, boolean initialized) {
    this.conn = conn;
    this.data = data;
    this.initialized = initialized;
    this.active = false;
    this.freed = false;
  }

  private PGXmlFactoryFactory getXmlFactoryFactory() throws SQLException {
    if (conn != null) {
      return conn.getXmlFactoryFactory();
    }
    return DefaultPGXmlFactoryFactory.INSTANCE;
  }

  @Override
  public void free() {
    try (ResourceLock ignore = lock.obtain()) {
      freed = true;
      data = null;
    }
  }

  @Override
  public @Nullable InputStream getBinaryStream() throws SQLException {
    try (ResourceLock ignore = lock.obtain()) {
      checkFreed();
      ensureInitialized();

      if (data == null) {
        return null;
      }

      try {
        return new ByteArrayInputStream(conn.getEncoding().encode(data));
      } catch (IOException ioe) {
        // This should be a can't happen exception. We just
        // decoded this data, so it would be surprising that
        // we couldn't encode it.
        // For this reason don't make it translatable.
        throw new PSQLException("Failed to re-encode xml data.", PSQLState.DATA_ERROR, ioe);
      }
    }
  }

  @Override
  public @Nullable Reader getCharacterStream() throws SQLException {
    try (ResourceLock ignore = lock.obtain()) {
      checkFreed();
      ensureInitialized();

      if (data == null) {
        return null;
      }

      return new StringReader(data);
    }
  }

  // We must implement this unsafely because that's what the
  // interface requires. Because it says we're returning T
  // which is unknown, none of the return values can satisfy it
  // as Java isn't going to understand the if statements that
  // ensure they are the same.
  //
  @Override
  public  @Nullable T getSource(@Nullable Class sourceClass)
      throws SQLException {
    try (ResourceLock ignore = lock.obtain()) {
      checkFreed();
      ensureInitialized();

      String data = this.data;
      if (data == null) {
        return null;
      }

      try {
        if (sourceClass == null || DOMSource.class.equals(sourceClass)) {
          DocumentBuilder builder = getXmlFactoryFactory().newDocumentBuilder();
          InputSource input = new InputSource(new StringReader(data));
          DOMSource domSource = new DOMSource(builder.parse(input));
          //noinspection unchecked
          return (T) domSource;
        } else if (SAXSource.class.equals(sourceClass)) {
          XMLReader reader = getXmlFactoryFactory().createXMLReader();
          InputSource is = new InputSource(new StringReader(data));
          return sourceClass.cast(new SAXSource(reader, is));
        } else if (StreamSource.class.equals(sourceClass)) {
          return sourceClass.cast(new StreamSource(new StringReader(data)));
        } else if (StAXSource.class.equals(sourceClass)) {
          XMLInputFactory xif = getXmlFactoryFactory().newXMLInputFactory();
          XMLStreamReader xsr = xif.createXMLStreamReader(new StringReader(data));
          return sourceClass.cast(new StAXSource(xsr));
        }
      } catch (Exception e) {
        throw new PSQLException(GT.tr("Unable to decode xml data."), PSQLState.DATA_ERROR, e);
      }

      throw new PSQLException(GT.tr("Unknown XML Source class: {0}", sourceClass),
          PSQLState.INVALID_PARAMETER_TYPE);
    }
  }

  @Override
  public @Nullable String getString() throws SQLException {
    try (ResourceLock ignore = lock.obtain()) {
      checkFreed();
      ensureInitialized();
      return data;
    }
  }

  @Override
  public OutputStream setBinaryStream() throws SQLException {
    try (ResourceLock ignore = lock.obtain()) {
      checkFreed();
      initialize();
      active = true;
      byteArrayOutputStream = new ByteArrayOutputStream();
      return byteArrayOutputStream;
    }
  }

  @Override
  public Writer setCharacterStream() throws SQLException {
    try (ResourceLock ignore = lock.obtain()) {
      checkFreed();
      initialize();
      active = true;
      stringWriter = new StringWriter();
      return stringWriter;
    }
  }

  @Override
  public  T setResult(Class resultClass) throws SQLException {
    try (ResourceLock ignore = lock.obtain()) {
      checkFreed();
      initialize();

      if (resultClass == null || DOMResult.class.equals(resultClass)) {
        domResult = new DOMResult();
        active = true;
        //noinspection unchecked
        return (T) domResult;
      } else if (SAXResult.class.equals(resultClass)) {
        try {
          SAXTransformerFactory transformerFactory = getXmlFactoryFactory().newSAXTransformerFactory();
          TransformerHandler transformerHandler = transformerFactory.newTransformerHandler();
          stringWriter = new StringWriter();
          transformerHandler.setResult(new StreamResult(stringWriter));
          active = true;
          return resultClass.cast(new SAXResult(transformerHandler));
        } catch (TransformerException te) {
          throw new PSQLException(GT.tr("Unable to create SAXResult for SQLXML."),
              PSQLState.UNEXPECTED_ERROR, te);
        }
      } else if (StreamResult.class.equals(resultClass)) {
        stringWriter = new StringWriter();
        active = true;
        return resultClass.cast(new StreamResult(stringWriter));
      } else if (StAXResult.class.equals(resultClass)) {
        StringWriter stringWriter = new StringWriter();
        this.stringWriter = stringWriter;
        try {
          XMLOutputFactory xof = getXmlFactoryFactory().newXMLOutputFactory();
          XMLStreamWriter xsw = xof.createXMLStreamWriter(stringWriter);
          active = true;
          return resultClass.cast(new StAXResult(xsw));
        } catch (XMLStreamException xse) {
          throw new PSQLException(GT.tr("Unable to create StAXResult for SQLXML"),
              PSQLState.UNEXPECTED_ERROR, xse);
        }
      }

      throw new PSQLException(GT.tr("Unknown XML Result class: {0}", resultClass),
          PSQLState.INVALID_PARAMETER_TYPE);
    }
  }

  @Override
  public void setString(String value) throws SQLException {
    try (ResourceLock ignore = lock.obtain()) {
      checkFreed();
      initialize();
      data = value;
    }
  }

  private void checkFreed() throws SQLException {
    if (freed) {
      throw new PSQLException(GT.tr("This SQLXML object has already been freed."),
          PSQLState.OBJECT_NOT_IN_STATE);
    }
  }

  private void ensureInitialized() throws SQLException {
    if (!initialized) {
      throw new PSQLException(
          GT.tr(
              "This SQLXML object has not been initialized, so you cannot retrieve data from it."),
          PSQLState.OBJECT_NOT_IN_STATE);
    }

    // Is anyone loading data into us at the moment?
    if (!active) {
      return;
    }

    if (byteArrayOutputStream != null) {
      try {
        data = conn.getEncoding().decode(byteArrayOutputStream.toByteArray());
      } catch (IOException ioe) {
        throw new PSQLException(GT.tr("Failed to convert binary xml data to encoding: {0}.",
            conn.getEncoding().name()), PSQLState.DATA_ERROR, ioe);
      } finally {
        byteArrayOutputStream = null;
        active = false;
      }
    } else if (stringWriter != null) {
      // This is also handling the work for Stream, SAX, and StAX Results
      // as they will use the same underlying stringwriter variable.
      //
      data = stringWriter.toString();
      stringWriter = null;
      active = false;
    } else if (domResult != null) {
      DOMResult domResult = this.domResult;
      // Copy the content from the result to a source
      // and use the identify transform to get it into a
      // friendlier result format.
      try {
        TransformerFactory factory = getXmlFactoryFactory().newTransformerFactory();
        Transformer transformer = factory.newTransformer();
        DOMSource domSource = new DOMSource(domResult.getNode());
        StringWriter stringWriter = new StringWriter();
        StreamResult streamResult = new StreamResult(stringWriter);
        transformer.transform(domSource, streamResult);
        data = stringWriter.toString();
      } catch (TransformerException te) {
        throw new PSQLException(GT.tr("Unable to convert DOMResult SQLXML data to a string."),
            PSQLState.DATA_ERROR, te);
      } finally {
        domResult = null;
        active = false;
      }
    }
  }

  private void initialize() throws SQLException {
    if (initialized) {
      throw new PSQLException(
          GT.tr(
              "This SQLXML object has already been initialized, so you cannot manipulate it further."),
          PSQLState.OBJECT_NOT_IN_STATE);
    }
    initialized = true;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy