Skip to content

Commit e533c5a

Browse files
authored
Merge pull request #131 from Tieqiong/fix_win
fix: win_py13 SystemError and MemoryError related to `redirect_stdout`
2 parents 6180000 + 864b183 commit e533c5a

File tree

7 files changed

+36
-58
lines changed

7 files changed

+36
-58
lines changed

news/fix_win.rst

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
**Added:**
2+
3+
* <news item>
4+
5+
**Changed:**
6+
7+
* <news item>
8+
9+
**Deprecated:**
10+
11+
* <news item>
12+
13+
**Removed:**
14+
15+
* Removed `restore_stdout` function and wrapper.
16+
17+
**Fixed:**
18+
19+
* Fixed `SystemError` and `MemoryError` for `redirect_stdout` on Windows with Python 3.13.
20+
21+
**Security:**
22+
23+
* <news item>

src/diffpy/pdffit2/output.py

-9
Original file line numberDiff line numberDiff line change
@@ -39,13 +39,4 @@ def redirect_stdout(dst):
3939
return
4040

4141

42-
def restore_stdout():
43-
"""Restore the standard output."""
44-
from diffpy.pdffit2.pdffit2 import restore_stdout
45-
46-
restore_stdout()
47-
global stdout
48-
return
49-
50-
5142
# End of file

src/extensions/pdffit2module/PyFileStreambuf.h

+9-4
Original file line numberDiff line numberDiff line change
@@ -67,16 +67,21 @@ class PyFileStreambuf : public std::streambuf
6767

6868
virtual int_type overflow( int_type c)
6969
{
70-
PyObject* rv;
71-
rv = PyObject_CallMethod(py_file, "write", "(s#)", &c, 1);
70+
char ch = static_cast<char>(c);
71+
PyObject* py_str = PyUnicode_FromStringAndSize(&ch, 1);
72+
if (!py_str) { return traits_type::eof(); }
73+
PyObject* rv = PyObject_CallMethod(py_file, "write", "O", py_str);
74+
Py_DECREF(py_str);
7275
if (rv) { Py_DECREF(rv); }
7376
return c;
7477
}
7578

7679
virtual std::streamsize xsputn(const char_type* s, std::streamsize n)
7780
{
78-
PyObject* rv;
79-
rv = PyObject_CallMethod(py_file, "write", "(s#)", s, n);
81+
PyObject* py_str = PyUnicode_DecodeUTF8(s, n, "replace");
82+
if (!py_str) { return 0; }
83+
PyObject* rv = PyObject_CallMethod(py_file, "write", "O", py_str);
84+
Py_DECREF(py_str);
8085
if (rv) { Py_DECREF(rv); }
8186
return n;
8287
}

src/extensions/pdffit2module/bindings.cc

-4
Original file line numberDiff line numberDiff line change
@@ -330,10 +330,6 @@ struct PyMethodDef pypdffit2_methods[] = {
330330
{pypdffit2_redirect_stdout__name__, pypdffit2_redirect_stdout,
331331
METH_VARARGS, pypdffit2_redirect_stdout__doc__},
332332

333-
//restore_stdout
334-
{pypdffit2_restore_stdout__name__, pypdffit2_restore_stdout,
335-
METH_VARARGS, pypdffit2_restore_stdout__doc__},
336-
337333
//is_element
338334
{pypdffit2_is_element__name__, pypdffit2_is_element,
339335
METH_VARARGS, pypdffit2_is_element__doc__},

src/extensions/pdffit2module/misc.cc

+4-35
Original file line numberDiff line numberDiff line change
@@ -2176,15 +2176,14 @@ char pypdffit2_redirect_stdout__name__[] = "redirect_stdout";
21762176
PyObject * pypdffit2_redirect_stdout(PyObject *, PyObject *args)
21772177
{
21782178
// instance of PyFileStreambuf which takes care of redirection
2179-
PyObject *py_file = 0;
2180-
int ok = PyArg_ParseTuple(args, "O", &py_file);
2181-
if (!ok) return 0;
2179+
PyObject *py_file = nullptr;
2180+
if (!PyArg_ParseTuple(args, "O", &py_file)) return nullptr;
21822181
// check if py_file has write and flush attributes
21832182
if ( !PyObject_HasAttrString(py_file, "write") ||
21842183
!PyObject_HasAttrString(py_file, "flush") )
21852184
{
21862185
PyErr_SetString(PyExc_TypeError, "expected file-like argument");
2187-
return 0;
2186+
return nullptr;
21882187
}
21892188
// create py_stdout_streambuf if necessary
21902189
if (!py_stdout_streambuf)
@@ -2195,38 +2194,8 @@ PyObject * pypdffit2_redirect_stdout(PyObject *, PyObject *args)
21952194
// on first redirection we need to assign new ostream to NS_PDFFIT2::pout
21962195
if (NS_PDFFIT2::pout == &std::cout)
21972196
{
2198-
NS_PDFFIT2::pout = new ostream(py_stdout_streambuf);
2199-
}
2200-
Py_INCREF(Py_None);
2201-
return Py_None;
2202-
}
2203-
2204-
// restore_stdout
2205-
char pypdffit2_restore_stdout__doc__[] =
2206-
"Restore engine output to the default stream (std::cout).";
2207-
char pypdffit2_restore_stdout__name__[] =
2208-
"restore_stdout";
2209-
2210-
PyObject * pypdffit2_restore_stdout(PyObject *, PyObject *args)
2211-
{
2212-
// no arguments.
2213-
if (!PyArg_ParseTuple(args, ""))
2214-
return 0;
2215-
2216-
// If the global output stream pointer is not std::cout, then delete the custom stream.
2217-
if (NS_PDFFIT2::pout != &std::cout)
2218-
{
2219-
delete NS_PDFFIT2::pout;
2220-
NS_PDFFIT2::pout = &std::cout;
2197+
NS_PDFFIT2::pout = new std::ostream(py_stdout_streambuf);
22212198
}
2222-
2223-
// Clean up the custom stream buffer
2224-
if (py_stdout_streambuf)
2225-
{
2226-
delete py_stdout_streambuf;
2227-
py_stdout_streambuf = nullptr;
2228-
}
2229-
22302199
Py_INCREF(Py_None);
22312200
return Py_None;
22322201
}

src/extensions/pdffit2module/misc.h

-5
Original file line numberDiff line numberDiff line change
@@ -477,11 +477,6 @@ extern char pypdffit2_redirect_stdout__name__[];
477477
extern "C"
478478
PyObject * pypdffit2_redirect_stdout(PyObject *, PyObject *);
479479

480-
// restore_stdout
481-
extern char pypdffit2_restore_stdout__doc__[];
482-
extern char pypdffit2_restore_stdout__name__[];
483-
extern "C"
484-
PyObject * pypdffit2_restore_stdout(PyObject *, PyObject *);
485480

486481
// is_element
487482
extern char pypdffit2_is_element__doc__[];

tests/conftest.py

-1
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,6 @@ def _capture(f, *args, **kwargs):
4545
f(*args, **kwargs)
4646
finally:
4747
diffpy.pdffit2.redirect_stdout(savestdout)
48-
diffpy.pdffit2.output.restore_stdout()
4948
return fp.getvalue()
5049

5150
return _capture

0 commit comments

Comments
 (0)