mirror of
https://github.com/PiBrewing/craftbeerpi4.git
synced 2025-01-01 10:21:44 +01:00
390 lines
13 KiB
Python
390 lines
13 KiB
Python
|
"""
|
||
|
Tests for the pandas.io.common functionalities
|
||
|
"""
|
||
|
from io import StringIO
|
||
|
import mmap
|
||
|
import os
|
||
|
from pathlib import Path
|
||
|
|
||
|
import pytest
|
||
|
|
||
|
from pandas.compat import is_platform_windows
|
||
|
import pandas.util._test_decorators as td
|
||
|
|
||
|
import pandas as pd
|
||
|
import pandas._testing as tm
|
||
|
|
||
|
import pandas.io.common as icom
|
||
|
|
||
|
|
||
|
class CustomFSPath:
|
||
|
"""For testing fspath on unknown objects"""
|
||
|
|
||
|
def __init__(self, path):
|
||
|
self.path = path
|
||
|
|
||
|
def __fspath__(self):
|
||
|
return self.path
|
||
|
|
||
|
|
||
|
# Functions that consume a string path and return a string or path-like object
|
||
|
path_types = [str, CustomFSPath, Path]
|
||
|
|
||
|
try:
|
||
|
from py.path import local as LocalPath
|
||
|
|
||
|
path_types.append(LocalPath)
|
||
|
except ImportError:
|
||
|
pass
|
||
|
|
||
|
HERE = os.path.abspath(os.path.dirname(__file__))
|
||
|
|
||
|
|
||
|
# https://github.com/cython/cython/issues/1720
|
||
|
@pytest.mark.filterwarnings("ignore:can't resolve package:ImportWarning")
|
||
|
class TestCommonIOCapabilities:
|
||
|
data1 = """index,A,B,C,D
|
||
|
foo,2,3,4,5
|
||
|
bar,7,8,9,10
|
||
|
baz,12,13,14,15
|
||
|
qux,12,13,14,15
|
||
|
foo2,12,13,14,15
|
||
|
bar2,12,13,14,15
|
||
|
"""
|
||
|
|
||
|
def test_expand_user(self):
|
||
|
filename = "~/sometest"
|
||
|
expanded_name = icom._expand_user(filename)
|
||
|
|
||
|
assert expanded_name != filename
|
||
|
assert os.path.isabs(expanded_name)
|
||
|
assert os.path.expanduser(filename) == expanded_name
|
||
|
|
||
|
def test_expand_user_normal_path(self):
|
||
|
filename = "/somefolder/sometest"
|
||
|
expanded_name = icom._expand_user(filename)
|
||
|
|
||
|
assert expanded_name == filename
|
||
|
assert os.path.expanduser(filename) == expanded_name
|
||
|
|
||
|
def test_stringify_path_pathlib(self):
|
||
|
rel_path = icom.stringify_path(Path("."))
|
||
|
assert rel_path == "."
|
||
|
redundant_path = icom.stringify_path(Path("foo//bar"))
|
||
|
assert redundant_path == os.path.join("foo", "bar")
|
||
|
|
||
|
@td.skip_if_no("py.path")
|
||
|
def test_stringify_path_localpath(self):
|
||
|
path = os.path.join("foo", "bar")
|
||
|
abs_path = os.path.abspath(path)
|
||
|
lpath = LocalPath(path)
|
||
|
assert icom.stringify_path(lpath) == abs_path
|
||
|
|
||
|
def test_stringify_path_fspath(self):
|
||
|
p = CustomFSPath("foo/bar.csv")
|
||
|
result = icom.stringify_path(p)
|
||
|
assert result == "foo/bar.csv"
|
||
|
|
||
|
@pytest.mark.parametrize(
|
||
|
"extension,expected",
|
||
|
[
|
||
|
("", None),
|
||
|
(".gz", "gzip"),
|
||
|
(".bz2", "bz2"),
|
||
|
(".zip", "zip"),
|
||
|
(".xz", "xz"),
|
||
|
(".GZ", "gzip"),
|
||
|
(".BZ2", "bz2"),
|
||
|
(".ZIP", "zip"),
|
||
|
(".XZ", "xz"),
|
||
|
],
|
||
|
)
|
||
|
@pytest.mark.parametrize("path_type", path_types)
|
||
|
def test_infer_compression_from_path(self, extension, expected, path_type):
|
||
|
path = path_type("foo/bar.csv" + extension)
|
||
|
compression = icom.infer_compression(path, compression="infer")
|
||
|
assert compression == expected
|
||
|
|
||
|
def test_get_filepath_or_buffer_with_path(self):
|
||
|
filename = "~/sometest"
|
||
|
filepath_or_buffer, _, _, should_close = icom.get_filepath_or_buffer(filename)
|
||
|
assert filepath_or_buffer != filename
|
||
|
assert os.path.isabs(filepath_or_buffer)
|
||
|
assert os.path.expanduser(filename) == filepath_or_buffer
|
||
|
assert not should_close
|
||
|
|
||
|
def test_get_filepath_or_buffer_with_buffer(self):
|
||
|
input_buffer = StringIO()
|
||
|
filepath_or_buffer, _, _, should_close = icom.get_filepath_or_buffer(
|
||
|
input_buffer
|
||
|
)
|
||
|
assert filepath_or_buffer == input_buffer
|
||
|
assert not should_close
|
||
|
|
||
|
def test_iterator(self):
|
||
|
reader = pd.read_csv(StringIO(self.data1), chunksize=1)
|
||
|
result = pd.concat(reader, ignore_index=True)
|
||
|
expected = pd.read_csv(StringIO(self.data1))
|
||
|
tm.assert_frame_equal(result, expected)
|
||
|
|
||
|
# GH12153
|
||
|
it = pd.read_csv(StringIO(self.data1), chunksize=1)
|
||
|
first = next(it)
|
||
|
tm.assert_frame_equal(first, expected.iloc[[0]])
|
||
|
tm.assert_frame_equal(pd.concat(it), expected.iloc[1:])
|
||
|
|
||
|
@pytest.mark.parametrize(
|
||
|
"reader, module, error_class, fn_ext",
|
||
|
[
|
||
|
(pd.read_csv, "os", FileNotFoundError, "csv"),
|
||
|
(pd.read_fwf, "os", FileNotFoundError, "txt"),
|
||
|
(pd.read_excel, "xlrd", FileNotFoundError, "xlsx"),
|
||
|
(pd.read_feather, "pyarrow", IOError, "feather"),
|
||
|
(pd.read_hdf, "tables", FileNotFoundError, "h5"),
|
||
|
(pd.read_stata, "os", FileNotFoundError, "dta"),
|
||
|
(pd.read_sas, "os", FileNotFoundError, "sas7bdat"),
|
||
|
(pd.read_json, "os", ValueError, "json"),
|
||
|
(pd.read_pickle, "os", FileNotFoundError, "pickle"),
|
||
|
],
|
||
|
)
|
||
|
def test_read_non_existent(self, reader, module, error_class, fn_ext):
|
||
|
pytest.importorskip(module)
|
||
|
|
||
|
path = os.path.join(HERE, "data", "does_not_exist." + fn_ext)
|
||
|
msg1 = fr"File (b')?.+does_not_exist\.{fn_ext}'? does not exist"
|
||
|
msg2 = fr"\[Errno 2\] No such file or directory: '.+does_not_exist\.{fn_ext}'"
|
||
|
msg3 = "Expected object or value"
|
||
|
msg4 = "path_or_buf needs to be a string file path or file-like"
|
||
|
msg5 = (
|
||
|
fr"\[Errno 2\] File .+does_not_exist\.{fn_ext} does not exist: "
|
||
|
fr"'.+does_not_exist\.{fn_ext}'"
|
||
|
)
|
||
|
msg6 = fr"\[Errno 2\] 没有那个文件或目录: '.+does_not_exist\.{fn_ext}'"
|
||
|
msg7 = (
|
||
|
fr"\[Errno 2\] File o directory non esistente: '.+does_not_exist\.{fn_ext}'"
|
||
|
)
|
||
|
msg8 = fr"Failed to open local file.+does_not_exist\.{fn_ext}"
|
||
|
|
||
|
with pytest.raises(
|
||
|
error_class,
|
||
|
match=fr"({msg1}|{msg2}|{msg3}|{msg4}|{msg5}|{msg6}|{msg7}|{msg8})",
|
||
|
):
|
||
|
reader(path)
|
||
|
|
||
|
@pytest.mark.parametrize(
|
||
|
"reader, module, error_class, fn_ext",
|
||
|
[
|
||
|
(pd.read_csv, "os", FileNotFoundError, "csv"),
|
||
|
(pd.read_table, "os", FileNotFoundError, "csv"),
|
||
|
(pd.read_fwf, "os", FileNotFoundError, "txt"),
|
||
|
(pd.read_excel, "xlrd", FileNotFoundError, "xlsx"),
|
||
|
(pd.read_feather, "pyarrow", IOError, "feather"),
|
||
|
(pd.read_hdf, "tables", FileNotFoundError, "h5"),
|
||
|
(pd.read_stata, "os", FileNotFoundError, "dta"),
|
||
|
(pd.read_sas, "os", FileNotFoundError, "sas7bdat"),
|
||
|
(pd.read_json, "os", ValueError, "json"),
|
||
|
(pd.read_pickle, "os", FileNotFoundError, "pickle"),
|
||
|
],
|
||
|
)
|
||
|
def test_read_expands_user_home_dir(
|
||
|
self, reader, module, error_class, fn_ext, monkeypatch
|
||
|
):
|
||
|
pytest.importorskip(module)
|
||
|
|
||
|
path = os.path.join("~", "does_not_exist." + fn_ext)
|
||
|
monkeypatch.setattr(icom, "_expand_user", lambda x: os.path.join("foo", x))
|
||
|
|
||
|
msg1 = fr"File (b')?.+does_not_exist\.{fn_ext}'? does not exist"
|
||
|
msg2 = fr"\[Errno 2\] No such file or directory: '.+does_not_exist\.{fn_ext}'"
|
||
|
msg3 = "Unexpected character found when decoding 'false'"
|
||
|
msg4 = "path_or_buf needs to be a string file path or file-like"
|
||
|
msg5 = (
|
||
|
fr"\[Errno 2\] File .+does_not_exist\.{fn_ext} does not exist: "
|
||
|
fr"'.+does_not_exist\.{fn_ext}'"
|
||
|
)
|
||
|
msg6 = fr"\[Errno 2\] 没有那个文件或目录: '.+does_not_exist\.{fn_ext}'"
|
||
|
msg7 = (
|
||
|
fr"\[Errno 2\] File o directory non esistente: '.+does_not_exist\.{fn_ext}'"
|
||
|
)
|
||
|
msg8 = fr"Failed to open local file.+does_not_exist\.{fn_ext}"
|
||
|
|
||
|
with pytest.raises(
|
||
|
error_class,
|
||
|
match=fr"({msg1}|{msg2}|{msg3}|{msg4}|{msg5}|{msg6}|{msg7}|{msg8})",
|
||
|
):
|
||
|
reader(path)
|
||
|
|
||
|
@pytest.mark.parametrize(
|
||
|
"reader, module, path",
|
||
|
[
|
||
|
(pd.read_csv, "os", ("io", "data", "csv", "iris.csv")),
|
||
|
(pd.read_table, "os", ("io", "data", "csv", "iris.csv")),
|
||
|
(
|
||
|
pd.read_fwf,
|
||
|
"os",
|
||
|
("io", "data", "fixed_width", "fixed_width_format.txt"),
|
||
|
),
|
||
|
(pd.read_excel, "xlrd", ("io", "data", "excel", "test1.xlsx")),
|
||
|
(
|
||
|
pd.read_feather,
|
||
|
"pyarrow",
|
||
|
("io", "data", "feather", "feather-0_3_1.feather"),
|
||
|
),
|
||
|
(
|
||
|
pd.read_hdf,
|
||
|
"tables",
|
||
|
("io", "data", "legacy_hdf", "datetimetz_object.h5"),
|
||
|
),
|
||
|
(pd.read_stata, "os", ("io", "data", "stata", "stata10_115.dta")),
|
||
|
(pd.read_sas, "os", ("io", "sas", "data", "test1.sas7bdat")),
|
||
|
(pd.read_json, "os", ("io", "json", "data", "tsframe_v012.json")),
|
||
|
(
|
||
|
pd.read_pickle,
|
||
|
"os",
|
||
|
("io", "data", "pickle", "categorical.0.25.0.pickle"),
|
||
|
),
|
||
|
],
|
||
|
)
|
||
|
@pytest.mark.filterwarnings(
|
||
|
"ignore:This method will be removed in future versions. "
|
||
|
r"Use 'tree.iter\(\)' or 'list\(tree.iter\(\)\)' instead."
|
||
|
":PendingDeprecationWarning"
|
||
|
)
|
||
|
def test_read_fspath_all(self, reader, module, path, datapath):
|
||
|
pytest.importorskip(module)
|
||
|
path = datapath(*path)
|
||
|
|
||
|
mypath = CustomFSPath(path)
|
||
|
result = reader(mypath)
|
||
|
expected = reader(path)
|
||
|
|
||
|
if path.endswith(".pickle"):
|
||
|
# categorical
|
||
|
tm.assert_categorical_equal(result, expected)
|
||
|
else:
|
||
|
tm.assert_frame_equal(result, expected)
|
||
|
|
||
|
@pytest.mark.parametrize(
|
||
|
"writer_name, writer_kwargs, module",
|
||
|
[
|
||
|
("to_csv", {}, "os"),
|
||
|
("to_excel", {"engine": "xlwt"}, "xlwt"),
|
||
|
("to_feather", {}, "pyarrow"),
|
||
|
("to_html", {}, "os"),
|
||
|
("to_json", {}, "os"),
|
||
|
("to_latex", {}, "os"),
|
||
|
("to_pickle", {}, "os"),
|
||
|
("to_stata", {"time_stamp": pd.to_datetime("2019-01-01 00:00")}, "os"),
|
||
|
],
|
||
|
)
|
||
|
def test_write_fspath_all(self, writer_name, writer_kwargs, module):
|
||
|
p1 = tm.ensure_clean("string")
|
||
|
p2 = tm.ensure_clean("fspath")
|
||
|
df = pd.DataFrame({"A": [1, 2]})
|
||
|
|
||
|
with p1 as string, p2 as fspath:
|
||
|
pytest.importorskip(module)
|
||
|
mypath = CustomFSPath(fspath)
|
||
|
writer = getattr(df, writer_name)
|
||
|
|
||
|
writer(string, **writer_kwargs)
|
||
|
with open(string, "rb") as f:
|
||
|
expected = f.read()
|
||
|
|
||
|
writer(mypath, **writer_kwargs)
|
||
|
with open(fspath, "rb") as f:
|
||
|
result = f.read()
|
||
|
|
||
|
assert result == expected
|
||
|
|
||
|
def test_write_fspath_hdf5(self):
|
||
|
# Same test as write_fspath_all, except HDF5 files aren't
|
||
|
# necessarily byte-for-byte identical for a given dataframe, so we'll
|
||
|
# have to read and compare equality
|
||
|
pytest.importorskip("tables")
|
||
|
|
||
|
df = pd.DataFrame({"A": [1, 2]})
|
||
|
p1 = tm.ensure_clean("string")
|
||
|
p2 = tm.ensure_clean("fspath")
|
||
|
|
||
|
with p1 as string, p2 as fspath:
|
||
|
mypath = CustomFSPath(fspath)
|
||
|
df.to_hdf(mypath, key="bar")
|
||
|
df.to_hdf(string, key="bar")
|
||
|
|
||
|
result = pd.read_hdf(fspath, key="bar")
|
||
|
expected = pd.read_hdf(string, key="bar")
|
||
|
|
||
|
tm.assert_frame_equal(result, expected)
|
||
|
|
||
|
|
||
|
@pytest.fixture
|
||
|
def mmap_file(datapath):
|
||
|
return datapath("io", "data", "csv", "test_mmap.csv")
|
||
|
|
||
|
|
||
|
class TestMMapWrapper:
|
||
|
def test_constructor_bad_file(self, mmap_file):
|
||
|
non_file = StringIO("I am not a file")
|
||
|
non_file.fileno = lambda: -1
|
||
|
|
||
|
# the error raised is different on Windows
|
||
|
if is_platform_windows():
|
||
|
msg = "The parameter is incorrect"
|
||
|
err = OSError
|
||
|
else:
|
||
|
msg = "[Errno 22]"
|
||
|
err = mmap.error
|
||
|
|
||
|
with pytest.raises(err, match=msg):
|
||
|
icom._MMapWrapper(non_file)
|
||
|
|
||
|
target = open(mmap_file, "r")
|
||
|
target.close()
|
||
|
|
||
|
msg = "I/O operation on closed file"
|
||
|
with pytest.raises(ValueError, match=msg):
|
||
|
icom._MMapWrapper(target)
|
||
|
|
||
|
def test_get_attr(self, mmap_file):
|
||
|
with open(mmap_file, "r") as target:
|
||
|
wrapper = icom._MMapWrapper(target)
|
||
|
|
||
|
attrs = dir(wrapper.mmap)
|
||
|
attrs = [attr for attr in attrs if not attr.startswith("__")]
|
||
|
attrs.append("__next__")
|
||
|
|
||
|
for attr in attrs:
|
||
|
assert hasattr(wrapper, attr)
|
||
|
|
||
|
assert not hasattr(wrapper, "foo")
|
||
|
|
||
|
def test_next(self, mmap_file):
|
||
|
with open(mmap_file, "r") as target:
|
||
|
wrapper = icom._MMapWrapper(target)
|
||
|
lines = target.readlines()
|
||
|
|
||
|
for line in lines:
|
||
|
next_line = next(wrapper)
|
||
|
assert next_line.strip() == line.strip()
|
||
|
|
||
|
with pytest.raises(StopIteration, match=r"^$"):
|
||
|
next(wrapper)
|
||
|
|
||
|
def test_unknown_engine(self):
|
||
|
with tm.ensure_clean() as path:
|
||
|
df = tm.makeDataFrame()
|
||
|
df.to_csv(path)
|
||
|
with pytest.raises(ValueError, match="Unknown engine"):
|
||
|
pd.read_csv(path, engine="pyt")
|
||
|
|
||
|
|
||
|
def test_is_fsspec_url():
|
||
|
assert icom.is_fsspec_url("gcs://pandas/somethingelse.com")
|
||
|
assert icom.is_fsspec_url("gs://pandas/somethingelse.com")
|
||
|
# the following is the only remote URL that is handled without fsspec
|
||
|
assert not icom.is_fsspec_url("http://pandas/somethingelse.com")
|
||
|
assert not icom.is_fsspec_url("random:pandas/somethingelse.com")
|
||
|
assert not icom.is_fsspec_url("/local/path")
|
||
|
assert not icom.is_fsspec_url("relative/local/path")
|