/*
 * Copyright (c) 2003-2013, John Wiegley.  All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are
 * met:
 *
 * - Redistributions of source code must retain the above copyright
 *   notice, this list of conditions and the following disclaimer.
 *
 * - 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.
 *
 * - Neither the name of New Artisans LLC 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
 * 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.
 */

#ifndef _PYFSTREAM_H
#define _PYFSTREAM_H

// pyofstream
// - a stream that writes on a Python file object

class pyoutbuf : public boost::noncopyable, public std::streambuf
{
  pyoutbuf();

protected:
  PyFileObject * fo;    // Python file object

public:
  // constructor
  pyoutbuf(PyFileObject * _fo) : fo(_fo) {
    TRACE_CTOR(pyoutbuf, "PyFileObject *");
  }
  ~pyoutbuf() throw() {
    TRACE_DTOR(pyoutbuf);
  }

protected:
  // write one character
  virtual int_type overflow (int_type c) {
    if (c != EOF) {
      char z[2];
      z[0] = static_cast<char>(c);
      z[1] = '\0';
      if (PyFile_WriteString(z, reinterpret_cast<PyObject *>(fo)) < 0) {
        return EOF;
      }
    }
    return c;
  }

  // write multiple characters
  virtual std::streamsize xsputn (const char* s, std::streamsize num) {
    char * buf = new char[num + 1];
    std::strncpy(buf, s, static_cast<std::size_t>(num));
    buf[num] = '\0';
    if (PyFile_WriteString(buf, reinterpret_cast<PyObject *>(fo)) < 0)
      num = 0;
    boost::checked_array_delete(buf);
    return num;
  }
};

class pyofstream : public boost::noncopyable, public std::ostream
{
  pyofstream();

protected:
  pyoutbuf buf;

public:
  pyofstream (PyFileObject * fo) : std::ostream(0), buf(fo) {
    rdbuf(&buf);
    TRACE_CTOR(pyofstream, "PyFileObject *");
  }
  ~pyofstream() throw() {
    TRACE_DTOR(pyofstream);
  }
};

// pyifstream
// - a stream that reads on a file descriptor

class pyinbuf : public boost::noncopyable, public std::streambuf
{
  pyinbuf();

protected:
  PyFileObject * fo;    // Python file object

protected:
  /* data buffer:
   * - at most, pbSize characters in putback area plus
   * - at most, bufSize characters in ordinary read buffer
   */
  static const size_t pbSize  = 4;    // size of putback area
  static const size_t bufSize = 1024; // size of the data buffer
  char buffer[bufSize + pbSize];      // data buffer

public:
  /* constructor
   * - initialize file descriptor
   * - initialize empty data buffer
   * - no putback area
   * => force underflow()
   */
  pyinbuf (PyFileObject * _fo) : fo(_fo) {
    setg (buffer+pbSize,     // beginning of putback area
          buffer+pbSize,     // read position
          buffer+pbSize);    // end position

    TRACE_CTOR(pyinbuf, "PyFileObject *");
  }
  ~pyinbuf() throw() {
    TRACE_DTOR(pyinbuf);
  }

protected:
  // insert new characters into the buffer
  virtual int_type underflow () {
#ifndef _MSC_VER
    using std::memmove;
#endif

    // is read position before end of buffer?
    if (gptr() < egptr()) {
      return traits_type::to_int_type(*gptr());
    }

    /* process size of putback area
     * - use number of characters read
     * - but at most size of putback area
     */
    size_t numPutback;
    numPutback = static_cast<size_t>(gptr() - eback());
    if (numPutback > pbSize) {
      numPutback = pbSize;
    }

    /* copy up to pbSize characters previously read into
     * the putback area
     */
    memmove (buffer+(pbSize-numPutback), gptr()-numPutback,
             numPutback);

    // read at most bufSize new characters
    PyObject *line = PyFile_GetLine(reinterpret_cast<PyObject *>(fo), bufSize);
    if (! line || ! PyString_Check(line)) {
      // ERROR or EOF
      return EOF;
    }

    Py_ssize_t num = PyString_Size(line);
    if (num == 0)
      return EOF;

    memmove(buffer+pbSize, PyString_AsString(line), static_cast<size_t>(num));

    // reset buffer pointers
    setg (buffer+(pbSize-numPutback),   // beginning of putback area
          buffer+pbSize,                // read position
          buffer+pbSize+num);           // end of buffer

    // return next character
    return traits_type::to_int_type(*gptr());
  }
};

class pyifstream : public boost::noncopyable, public std::istream
{
  pyifstream();

protected:
  pyinbuf buf;

public:
  pyifstream (PyFileObject * fo) : std::istream(0), buf(fo) {
    rdbuf(&buf);
    TRACE_CTOR(pyifstream, "PyFileObject *");
  }
  ~pyifstream() throw() {
    TRACE_DTOR(pyifstream);
  }
};

#endif // _PYFSTREAM_H