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

org.exist.util.BinaryValueInputSource Maven / Gradle / Ivy

There is a newer version: 6.3.0
Show newest version
/*
 * eXist-db Open Source Native XML Database
 * Copyright (C) 2001 The eXist-db Authors
 *
 * [email protected]
 * http://www.exist-db.org
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 */
package org.exist.util;

import org.apache.commons.io.output.CountingOutputStream;
import org.apache.commons.io.output.NullOutputStream;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.exist.xquery.value.BinaryValue;

import java.io.IOException;
import java.io.InputStream;
import java.io.Reader;
import java.util.Optional;

/**
 * Input Source for a Binary Value.
 *
 * @author Adam Retter
 */
public class BinaryValueInputSource extends EXistInputSource {
    private final static Logger LOG = LogManager.getLogger(BinaryValueInputSource.class);

    private Optional binaryValue = Optional.empty();
    private Optional inputStream = Optional.empty();
    private long length = -1;

    /**
     * Constructor which calls {@link #setBinaryValue(BinaryValue)}
     * @param binaryValue
     * The binaryValue passed to {@link #setBinaryValue(BinaryValue)}
     */
    public BinaryValueInputSource(final BinaryValue binaryValue) {
        super();
        setBinaryValue(binaryValue);
    }

    /**
     * If a binary file source has been set, the BinaryValue
     * object used for that is returned
     *
     * @return The BinaryValue object.
     */
    public BinaryValue getBinaryValue() {
        return binaryValue.orElse(null);
    }

    /**
     * This method sets the BinaryValue object
     *
     * @param binaryValue The BinaryValue.
     *
     * @throws IllegalStateException if the InputSource was previously closed
     */
    public void setBinaryValue(final BinaryValue binaryValue) {
        assertOpen();

        close();
        this.binaryValue = Optional.of(binaryValue);
        reOpen();
    }

    /**
     * This method was re-implemented to open a
     * new InputStream each time it is called.
     *
     * @return If the binaryvalue was set, and it could be opened, an InputStream object.
     * null, otherwise.
     *
     * @throws IllegalStateException if the InputSource was previously closed
     */
    @Override
    public InputStream getByteStream() {
        assertOpen();

        // close any open stream first
        close();

        if (binaryValue.isPresent()) {
            this.inputStream = Optional.of(binaryValue.get().getInputStream());
            reOpen();
            return inputStream.get();
        }

        return null;
    }

    /**
     * This method now does nothing, so collateral
     * effects from superclass with this one are avoided
     *
     * @throws IllegalStateException if the InputSource was previously closed
     */
    @Override
    public void setByteStream(final InputStream is) {
        assertOpen();
        // Nothing, so collateral effects are avoided!
    }

    /**
     * This method now does nothing, so collateral
     * effects from superclass with this one are avoided
     *
     * @throws IllegalStateException if the InputSource was previously closed
     */
    @Override
    public void setCharacterStream(final Reader r) {
        assertOpen();
        // Nothing, so collateral effects are avoided!
    }

    /**
     * This method now does nothing, so collateral
     * effects from superclass with this one are avoided
     *
     * @throws IllegalStateException if the InputSource was previously closed
     */
    @Override
    public void setSystemId(final String systemId) {
        assertOpen();
        // Nothing, so collateral effects are avoided!
    }

    /**
     * @see EXistInputSource#getByteStreamLength()
     *
     * @throws IllegalStateException if the InputSource was previously closed
     */
    @Override
    public long getByteStreamLength() {
        assertOpen();
        if (length == -1 && binaryValue.isPresent()) {
            try (final CountingOutputStream cos = new CountingOutputStream(NullOutputStream.NULL_OUTPUT_STREAM)) {
                binaryValue.get().streamBinaryTo(cos);
                length = cos.getByteCount();
            } catch(final IOException e) {
                LOG.error(e);
            }
        }

        return length;
    }

    /**
     * @see EXistInputSource#getSymbolicPath()
     *
     * @throws IllegalStateException if the InputSource was previously closed
     */
    @Override
    public String getSymbolicPath() {
        assertOpen();
        return null;
    }

    @Override
    public void close() {
        if(!isClosed()) {
            try {
                if (inputStream.isPresent()) {
                    try {
                        inputStream.get().close();
                    } catch (final IOException e) {
                        LOG.warn(e);
                    }
                    inputStream = Optional.empty();
                    length = -1;
                }
            } finally {
                super.close();
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy