/**
 * @copyright
 * ====================================================================
 * Copyright (c) 2003-2004 CollabNet.  All rights reserved.
 *
 * This software is licensed as described in the file COPYING, which
 * you should have received as part of this distribution.  The terms
 * are also available at http://subversion.tigris.org/license-1.html.
 * If newer versions of this license are posted there, you may use a
 * newer version instead, at your option.
 *
 * This software consists of voluntary contributions made by many
 * individuals.  For exact contribution history, see the revision
 * history and logs, available at http://subversion.tigris.org/.
 * ====================================================================
 * @endcopyright
 *
 * @file Outputer.cpp
 * @brief Implementation of the class Outputer
 */

#include "Outputer.h"
#include "JNIUtil.h"
#include "JNIByteArray.h"

/**
 * create an Outputer object
 * @param jthis the java object to be stored
 */
Outputer::Outputer(jobject jthis)
{
    m_jthis = jthis;
}

/**
 * destroy an Inputer object
 */
Outputer::~Outputer()
{
    // the m_jthis does not need to be destroyed, because it is the passed
    // in parameter to the java method.

}

/**
 * create a svn_stream_t structure for this object. This will be used as an
 * output stream by subversion
 * @param pool  the pool, from which the structure is allocated
 * @return the output stream
 */
svn_stream_t *Outputer::getStream(const Pool & pool)
{
    // create a stream with this as the baton and set the write and close
    // functions
    svn_stream_t *ret = svn_stream_create(this, pool.pool());
    svn_stream_set_write(ret, Outputer::write);
    svn_stream_set_close(ret, Outputer::close);
    return ret;
}

/**
 * implements svn_write_fn_t to write data out from subversion
 * @param baton     an Outputer object for the callback
 * @param buffer    the buffer for the write data
 * @param len       on input the buffer len, on output the number of written 
 *                  bytes
 * @return a subversion error or SVN_NO_ERROR
 */
svn_error_t *Outputer::write(void *baton, const char *buffer, apr_size_t *len)
{
    JNIEnv *env = JNIUtil::getEnv();
    // an object of our class is passed in as the baton
    Outputer *that = (Outputer*)baton;

    // the method id will not change during
    // the time this library is loaded, so
    // it can be cached.
   static jmethodID mid = 0;
    if(mid == 0)
    {
        jclass clazz = env->FindClass(JAVA_PACKAGE"/OutputInterface");
        if(JNIUtil::isJavaExceptionThrown())
        {
            return SVN_NO_ERROR;
        }
        mid = env->GetMethodID(clazz, "write", "([B)I");
        if(JNIUtil::isJavaExceptionThrown() || mid == 0)
        {
            return SVN_NO_ERROR;
        }
        env->DeleteLocalRef(clazz);
        if(JNIUtil::isJavaExceptionThrown())
        {
            return SVN_NO_ERROR;
        }
    }

    // convert the data to a java byte array
    jbyteArray data = JNIUtil::makeJByteArray((const signed char*)buffer, *len);
    if(JNIUtil::isJavaExceptionThrown())
    {
        return SVN_NO_ERROR;
    }

    // write the data
    jint written = env->CallIntMethod(that->m_jthis, mid, data);
    if(JNIUtil::isJavaExceptionThrown())
    {
        return SVN_NO_ERROR;
    }

    // return the number of bytes written
    *len = written;

    return SVN_NO_ERROR;
}
/**
 * implements svn_close_fn_t to close the output stream
 * @param baton     an Outputer object for the callback
 * @return a subversion error or SVN_NO_ERROR
 */
svn_error_t *Outputer::close(void *baton)
{
    JNIEnv *env = JNIUtil::getEnv();
    // an object of our class is passed in as the baton
    Outputer *that = (Outputer*)baton;

    // the method id will not change during
    // the time this library is loaded, so
    // it can be cached.
    static jmethodID mid = 0;
    if(mid == 0)
    {
        jclass clazz = env->FindClass(JAVA_PACKAGE"/OutputInterface");
        if(JNIUtil::isJavaExceptionThrown())
        {
            return SVN_NO_ERROR;
        }
        mid = env->GetMethodID(clazz, "close", "()V");
        if(JNIUtil::isJavaExceptionThrown() || mid == 0)
        {
            return SVN_NO_ERROR;
        }
        env->DeleteLocalRef(clazz);
        if(JNIUtil::isJavaExceptionThrown())
        {
            return SVN_NO_ERROR;
        }
    }

    // call the java object, to close the stream
    env->CallVoidMethod(that->m_jthis, mid);
    if(JNIUtil::isJavaExceptionThrown())
    {
        return SVN_NO_ERROR;
    }

    return SVN_NO_ERROR;
}


syntax highlighted by Code2HTML, v. 0.9.1