|
|
# encoding: utf-8
|
|
|
"""Tests for io.py"""
|
|
|
|
|
|
# Copyright (c) IPython Development Team.
|
|
|
# Distributed under the terms of the Modified BSD License.
|
|
|
|
|
|
from __future__ import print_function
|
|
|
from __future__ import absolute_import
|
|
|
|
|
|
import io as stdlib_io
|
|
|
import os.path
|
|
|
import stat
|
|
|
import sys
|
|
|
|
|
|
from subprocess import Popen, PIPE
|
|
|
import unittest
|
|
|
|
|
|
import nose.tools as nt
|
|
|
|
|
|
from IPython.testing.decorators import skipif, skip_win32
|
|
|
from IPython.utils.io import (Tee, capture_output, unicode_std_stream,
|
|
|
atomic_writing,
|
|
|
)
|
|
|
from IPython.utils.py3compat import doctest_refactor_print, PY3
|
|
|
from IPython.utils.tempdir import TemporaryDirectory
|
|
|
|
|
|
if PY3:
|
|
|
from io import StringIO
|
|
|
else:
|
|
|
from StringIO import StringIO
|
|
|
|
|
|
|
|
|
def test_tee_simple():
|
|
|
"Very simple check with stdout only"
|
|
|
chan = StringIO()
|
|
|
text = 'Hello'
|
|
|
tee = Tee(chan, channel='stdout')
|
|
|
print(text, file=chan)
|
|
|
nt.assert_equal(chan.getvalue(), text+"\n")
|
|
|
|
|
|
|
|
|
class TeeTestCase(unittest.TestCase):
|
|
|
|
|
|
def tchan(self, channel, check='close'):
|
|
|
trap = StringIO()
|
|
|
chan = StringIO()
|
|
|
text = 'Hello'
|
|
|
|
|
|
std_ori = getattr(sys, channel)
|
|
|
setattr(sys, channel, trap)
|
|
|
|
|
|
tee = Tee(chan, channel=channel)
|
|
|
print(text, end='', file=chan)
|
|
|
setattr(sys, channel, std_ori)
|
|
|
trap_val = trap.getvalue()
|
|
|
nt.assert_equal(chan.getvalue(), text)
|
|
|
if check=='close':
|
|
|
tee.close()
|
|
|
else:
|
|
|
del tee
|
|
|
|
|
|
def test(self):
|
|
|
for chan in ['stdout', 'stderr']:
|
|
|
for check in ['close', 'del']:
|
|
|
self.tchan(chan, check)
|
|
|
|
|
|
def test_io_init():
|
|
|
"""Test that io.stdin/out/err exist at startup"""
|
|
|
for name in ('stdin', 'stdout', 'stderr'):
|
|
|
cmd = doctest_refactor_print("from IPython.utils import io;print io.%s.__class__"%name)
|
|
|
p = Popen([sys.executable, '-c', cmd],
|
|
|
stdout=PIPE)
|
|
|
p.wait()
|
|
|
classname = p.stdout.read().strip().decode('ascii')
|
|
|
# __class__ is a reference to the class object in Python 3, so we can't
|
|
|
# just test for string equality.
|
|
|
assert 'IPython.utils.io.IOStream' in classname, classname
|
|
|
|
|
|
def test_capture_output():
|
|
|
"""capture_output() context works"""
|
|
|
|
|
|
with capture_output() as io:
|
|
|
print('hi, stdout')
|
|
|
print('hi, stderr', file=sys.stderr)
|
|
|
|
|
|
nt.assert_equal(io.stdout, 'hi, stdout\n')
|
|
|
nt.assert_equal(io.stderr, 'hi, stderr\n')
|
|
|
|
|
|
|
|
|
def test_atomic_writing():
|
|
|
class CustomExc(Exception): pass
|
|
|
|
|
|
with TemporaryDirectory() as td:
|
|
|
f1 = os.path.join(td, 'penguin')
|
|
|
with stdlib_io.open(f1, 'w') as f:
|
|
|
f.write(u'Before')
|
|
|
|
|
|
if os.name != 'nt':
|
|
|
os.chmod(f1, 0o701)
|
|
|
orig_mode = stat.S_IMODE(os.stat(f1).st_mode)
|
|
|
|
|
|
f2 = os.path.join(td, 'flamingo')
|
|
|
try:
|
|
|
os.symlink(f1, f2)
|
|
|
have_symlink = True
|
|
|
except (AttributeError, NotImplementedError, OSError):
|
|
|
# AttributeError: Python doesn't support it
|
|
|
# NotImplementedError: The system doesn't support it
|
|
|
# OSError: The user lacks the privilege (Windows)
|
|
|
have_symlink = False
|
|
|
|
|
|
with nt.assert_raises(CustomExc):
|
|
|
with atomic_writing(f1) as f:
|
|
|
f.write(u'Failing write')
|
|
|
raise CustomExc
|
|
|
|
|
|
# Because of the exception, the file should not have been modified
|
|
|
with stdlib_io.open(f1, 'r') as f:
|
|
|
nt.assert_equal(f.read(), u'Before')
|
|
|
|
|
|
with atomic_writing(f1) as f:
|
|
|
f.write(u'Overwritten')
|
|
|
|
|
|
with stdlib_io.open(f1, 'r') as f:
|
|
|
nt.assert_equal(f.read(), u'Overwritten')
|
|
|
|
|
|
if os.name != 'nt':
|
|
|
mode = stat.S_IMODE(os.stat(f1).st_mode)
|
|
|
nt.assert_equal(mode, orig_mode)
|
|
|
|
|
|
if have_symlink:
|
|
|
# Check that writing over a file preserves a symlink
|
|
|
with atomic_writing(f2) as f:
|
|
|
f.write(u'written from symlink')
|
|
|
|
|
|
with stdlib_io.open(f1, 'r') as f:
|
|
|
nt.assert_equal(f.read(), u'written from symlink')
|
|
|
|
|
|
def _save_umask():
|
|
|
global umask
|
|
|
umask = os.umask(0)
|
|
|
os.umask(umask)
|
|
|
|
|
|
def _restore_umask():
|
|
|
os.umask(umask)
|
|
|
|
|
|
@skip_win32
|
|
|
@nt.with_setup(_save_umask, _restore_umask)
|
|
|
def test_atomic_writing_umask():
|
|
|
with TemporaryDirectory() as td:
|
|
|
os.umask(0o022)
|
|
|
f1 = os.path.join(td, '1')
|
|
|
with atomic_writing(f1) as f:
|
|
|
f.write(u'1')
|
|
|
mode = stat.S_IMODE(os.stat(f1).st_mode)
|
|
|
nt.assert_equal(mode, 0o644, '{:o} != 644'.format(mode))
|
|
|
|
|
|
os.umask(0o057)
|
|
|
f2 = os.path.join(td, '2')
|
|
|
with atomic_writing(f2) as f:
|
|
|
f.write(u'2')
|
|
|
mode = stat.S_IMODE(os.stat(f2).st_mode)
|
|
|
nt.assert_equal(mode, 0o620, '{:o} != 620'.format(mode))
|
|
|
|
|
|
|
|
|
def test_atomic_writing_newlines():
|
|
|
with TemporaryDirectory() as td:
|
|
|
path = os.path.join(td, 'testfile')
|
|
|
|
|
|
lf = u'a\nb\nc\n'
|
|
|
plat = lf.replace(u'\n', os.linesep)
|
|
|
crlf = lf.replace(u'\n', u'\r\n')
|
|
|
|
|
|
# test default
|
|
|
with stdlib_io.open(path, 'w') as f:
|
|
|
f.write(lf)
|
|
|
with stdlib_io.open(path, 'r', newline='') as f:
|
|
|
read = f.read()
|
|
|
nt.assert_equal(read, plat)
|
|
|
|
|
|
# test newline=LF
|
|
|
with stdlib_io.open(path, 'w', newline='\n') as f:
|
|
|
f.write(lf)
|
|
|
with stdlib_io.open(path, 'r', newline='') as f:
|
|
|
read = f.read()
|
|
|
nt.assert_equal(read, lf)
|
|
|
|
|
|
# test newline=CRLF
|
|
|
with atomic_writing(path, newline='\r\n') as f:
|
|
|
f.write(lf)
|
|
|
with stdlib_io.open(path, 'r', newline='') as f:
|
|
|
read = f.read()
|
|
|
nt.assert_equal(read, crlf)
|
|
|
|
|
|
# test newline=no convert
|
|
|
text = u'crlf\r\ncr\rlf\n'
|
|
|
with atomic_writing(path, newline='') as f:
|
|
|
f.write(text)
|
|
|
with stdlib_io.open(path, 'r', newline='') as f:
|
|
|
read = f.read()
|
|
|
nt.assert_equal(read, text)
|
|
|
|