boto3
我有一种使用(1.4.4)、pyarrow
(0.4.1) 和pandas
(0.20.3)来实现这一目标的 hacky 方法。
首先,我可以像这样在本地读取单个 parquet 文件:
import pyarrow.parquet as pq
path = 'parquet/part-r-00000-1e638be4-e31f-498a-a359-47d017a0059c.gz.parquet'
table = pq.read_table(path)
df = table.to_pandas()
我还可以像这样在本地读取镶木地板文件的目录:
import pyarrow.parquet as pq
dataset = pq.ParquetDataset('parquet/')
table = dataset.read()
df = table.to_pandas()
两者都像魅力一样工作。现在我想用存储在 S3 存储桶中的文件远程实现相同的目标。我希望这样的事情会起作用:
dataset = pq.ParquetDataset('s3n://dsn/to/my/bucket')
但它没有:
OSError: Passed non-file path: s3n://dsn/to/my/bucket
在彻底阅读了 pyarrow 的文档之后,目前这似乎是不可能的。所以我提出了以下解决方案:
从 S3 读取单个文件并获取 pandas 数据框:
import io
import boto3
import pyarrow.parquet as pq
buffer = io.BytesIO()
s3 = boto3.resource('s3')
s3_object = s3.Object('bucket-name', 'key/to/parquet/file.gz.parquet')
s3_object.download_fileobj(buffer)
table = pq.read_table(buffer)
df = table.to_pandas()
这里是我从 S3 文件夹路径创建 pandas 数据框的 hacky、未优化的解决方案:
import io
import boto3
import pandas as pd
import pyarrow.parquet as pq
bucket_name = 'bucket-name'
def download_s3_parquet_file(s3, bucket, key):
buffer = io.BytesIO()
s3.Object(bucket, key).download_fileobj(buffer)
return buffer
client = boto3.client('s3')
s3 = boto3.resource('s3')
objects_dict = client.list_objects_v2(Bucket=bucket_name, Prefix='my/folder/prefix')
s3_keys = [item['Key'] for item in objects_dict['Contents'] if item['Key'].endswith('.parquet')]
buffers = [download_s3_parquet_file(s3, bucket_name, key) for key in s3_keys]
dfs = [pq.read_table(buffer).to_pandas() for buffer in buffers]
df = pd.concat(dfs, ignore_index=True)
有没有更好的方法来实现这一目标?也许某种使用pyarrow的熊猫连接器?我想避免使用pyspark
,但如果没有其他解决方案,那么我会接受。