Перейти к основному содержанию
Перейти к основному содержанию

Как выполнять запросы к данным в бакете S3

Значительная часть мировых данных хранится в бакетах Amazon S3. В этом руководстве вы узнаете, как выполнять запросы к этим данным с помощью chDB.

Настройка

Сначала создайте виртуальное окружение:

python -m venv .venv
source .venv/bin/activate

Теперь установите chDB. Убедитесь, что у вас установлена версия не ниже 2.0.2:

pip install "chdb>=2.0.2"

Теперь установим IPython:

pip install ipython

Мы будем использовать ipython для выполнения команд далее в этом руководстве, который можно запустить следующей командой:

ipython

Вы также можете использовать код в скрипте Python или в вашем любимом ноутбуке (например, Jupyter Notebook).

Получение списка файлов в бакете S3

Начнём с получения списка всех файлов в бакете S3, который содержит отзывы Amazon. Для этого мы можем использовать табличную функцию s3 и передать путь к файлу или шаблон (wildcard) для набора файлов.

Совет

Если передать только имя бакета, будет сгенерировано исключение.

Мы также будем использовать формат ввода One, чтобы файл не парсился: вместо этого по каждому файлу будет возвращена одна строка, и мы сможем получить доступ к файлу через виртуальный столбец _file, а к пути — через виртуальный столбец _path.

import chdb

chdb.query("""
SELECT
    _file,
    _path
FROM s3('s3://datasets-documentation/amazon_reviews/*.parquet', One)
SETTINGS output_format_pretty_row_numbers=0
""", 'PrettyCompact')
┌─_file───────────────────────────────┬─_path─────────────────────────────────────────────────────────────────────┐
│ amazon_reviews_2010.snappy.parquet  │ datasets-documentation/amazon_reviews/amazon_reviews_2010.snappy.parquet  │
│ amazon_reviews_1990s.snappy.parquet │ datasets-documentation/amazon_reviews/amazon_reviews_1990s.snappy.parquet │
│ amazon_reviews_2013.snappy.parquet  │ datasets-documentation/amazon_reviews/amazon_reviews_2013.snappy.parquet  │
│ amazon_reviews_2015.snappy.parquet  │ datasets-documentation/amazon_reviews/amazon_reviews_2015.snappy.parquet  │
│ amazon_reviews_2014.snappy.parquet  │ datasets-documentation/amazon_reviews/amazon_reviews_2014.snappy.parquet  │
│ amazon_reviews_2012.snappy.parquet  │ datasets-documentation/amazon_reviews/amazon_reviews_2012.snappy.parquet  │
│ amazon_reviews_2000s.snappy.parquet │ datasets-documentation/amazon_reviews/amazon_reviews_2000s.snappy.parquet │
│ amazon_reviews_2011.snappy.parquet  │ datasets-documentation/amazon_reviews/amazon_reviews_2011.snappy.parquet  │
└─────────────────────────────────────┴───────────────────────────────────────────────────────────────────────────┘

Этот бакет содержит только файлы Parquet.

Выполнение запросов к файлам в бакете S3

Теперь давайте узнаем, как выполнять запросы к этим файлам. Если мы хотим посчитать количество строк в каждом из этих файлов, мы можем выполнить следующий запрос:

chdb.query("""
SELECT
    _file,
    count() AS count,
    formatReadableQuantity(count) AS readableCount    
FROM s3('s3://datasets-documentation/amazon_reviews/*.parquet')
GROUP BY ALL
SETTINGS output_format_pretty_row_numbers=0
""", 'PrettyCompact')
┌─_file───────────────────────────────┬────count─┬─readableCount───┐
│ amazon_reviews_2013.snappy.parquet  │ 28034255 │ 28,03 млн       │
│ amazon_reviews_1990s.snappy.parquet │   639532 │ 639,53 тыс.     │
│ amazon_reviews_2011.snappy.parquet  │  6112495 │ 6,11 млн        │
│ amazon_reviews_2015.snappy.parquet  │ 41905631 │ 41,91 млн       │
│ amazon_reviews_2012.snappy.parquet  │ 11541011 │ 11,54 млн       │
│ amazon_reviews_2000s.snappy.parquet │ 14728295 │ 14,73 млн       │
│ amazon_reviews_2014.snappy.parquet  │ 44127569 │ 44,13 млн       │
│ amazon_reviews_2010.snappy.parquet  │  3868472 │ 3,87 млн        │
└─────────────────────────────────────┴──────────┴─────────────────┘

Мы также можем передать HTTP-URI бакета S3 и получим те же результаты:

chdb.query("""
SELECT
    _file,
    count() AS count,
    formatReadableQuantity(count) AS readableCount    
FROM s3('https://datasets-documentation.s3.eu-west-3.amazonaws.com/amazon_reviews/*.parquet')
GROUP BY ALL
SETTINGS output_format_pretty_row_numbers=0
""", 'PrettyCompact')

Давайте рассмотрим схему этих файлов Parquet с помощью оператора DESCRIBE:

chdb.query("""
DESCRIBE s3('s3://datasets-documentation/amazon_reviews/*.parquet')
SETTINGS describe_compact_output=1
""", 'PrettyCompact')
    ┌─name──────────────┬─type─────────────┐
 1. │ review_date       │ Nullable(UInt16) │
 2. │ marketplace       │ Nullable(String) │
 3. │ customer_id       │ Nullable(UInt64) │
 4. │ review_id         │ Nullable(String) │
 5. │ product_id        │ Nullable(String) │
 6. │ product_parent    │ Nullable(UInt64) │
 7. │ product_title     │ Nullable(String) │
 8. │ product_category  │ Nullable(String) │
 9. │ star_rating       │ Nullable(UInt8)  │
10. │ helpful_votes     │ Nullable(UInt32) │
11. │ total_votes       │ Nullable(UInt32) │
12. │ vine              │ Nullable(Bool)   │
13. │ verified_purchase │ Nullable(Bool)   │
14. │ review_headline   │ Nullable(String) │
15. │ review_body       │ Nullable(String) │
    └───────────────────┴──────────────────┘

Теперь вычислим категории товаров с наибольшим числом отзывов, а также среднюю оценку в звёздах:

chdb.query("""
SELECT product_category, count() AS reviews, round(avg(star_rating), 2) as avg
FROM s3('s3://datasets-documentation/amazon_reviews/*.parquet')
GROUP BY ALL
LIMIT 10
""", 'PrettyCompact')
    ┌─product_category─┬──reviews─┬──avg─┐
 1. │ Игрушки          │  4864056 │ 4.21 │
 2. │ Одежда           │  5906085 │ 4.11 │
 3. │ Багаж            │   348644 │ 4.22 │
 4. │ Кухня            │  4880297 │ 4.21 │
 5. │ Книги            │ 19530930 │ 4.34 │
 6. │ Товары для отдыха│  2302327 │ 4.24 │
 7. │ Видео            │   380596 │ 4.19 │
 8. │ Продукты         │  2402365 │ 4.31 │
 9. │ Обувь            │  4366757 │ 4.24 │
10. │ Ювелирные изделия│  1767667 │ 4.14 │
    └──────────────────┴──────────┴──────┘

Выполнение запросов к файлам в приватном бакете S3

Если мы выполняем запросы к файлам в приватном бакете S3, необходимо передать ключ доступа и секретный ключ. Эти учетные данные можно передать в табличную функцию s3:

chdb.query("""
SELECT product_category, count() AS reviews, round(avg(star_rating), 2) as avg
FROM s3('s3://datasets-documentation/amazon_reviews/*.parquet', 'access-key', 'secret')
GROUP BY ALL
LIMIT 10
""", 'PrettyCompact')
Примечание

Этот запрос не будет работать, потому что это публичный бакет!

Альтернативный способ — использовать именованные коллекции, но этот подход пока не поддерживается в chDB.