9.2 KiB
slug | sidebar_position | sidebar_label |
---|---|---|
/ja/engines/table-engines/special/executable | 40 | Executable |
Executable および ExecutablePool テーブルエンジン
Executable
および ExecutablePool
テーブルエンジンを使用すると、stdout に行を書き込むことによって定義したスクリプトから行を生成するテーブルを定義できます。実行スクリプトは users_scripts
ディレクトリに保存され、任意のソースからデータを読み取ることができます。
Executable
テーブル: スクリプトは各クエリで実行されますExecutablePool
テーブル: プール内で永続的なプロセスを維持し、読み取りのためにプールからプロセスを取得します
任意で 1 つ以上の入力クエリを含め、それらの結果を stdin にストリームしてスクリプトが読み取れるようにすることができます。
Executable テーブルの作成
Executable
テーブルエンジンは、スクリプトの名前と受信データのフォーマットの 2 つのパラメータを必要とします。任意で 1 つ以上の入力クエリを渡すことができます:
Executable(script_name, format, [input_query...])
以下は、Executable
テーブルに関する設定です:
send_chunk_header
- 説明: チャンク処理の前に各チャンクの行数を送信します。この設定によってリソースの一部を事前に割り当てる脚本を書くのに役立ちます
- デフォルト値: false
command_termination_timeout
- 説明: コマンド終了タイムアウト(秒単位)
- デフォルト値: 10
command_read_timeout
- 説明: コマンド stdout からデータを読み取るためのタイムアウト(ミリ秒単位)
- デフォルト値: 10000
command_write_timeout
- 説明: コマンド stdin にデータを書き込むためのタイムアウト(ミリ秒単位)
- デフォルト値: 10000
例を見てみましょう。この Python スクリプトは my_script.py
と名付けられ、user_scripts
フォルダに保存されています。それは数値 i
を読み取り、各文字列がタブで区切られた番号とともに、i
個のランダムな文字列を出力します:
#!/usr/bin/python3
import sys
import string
import random
def main():
# 入力値を読み取る
for number in sys.stdin:
i = int(number)
# ランダムな行を生成する
for id in range(0, i):
letters = string.ascii_letters
random_string = ''.join(random.choices(letters ,k=10))
print(str(id) + '\t' + random_string + '\n', end='')
# 結果を stdout にフラッシュする
sys.stdout.flush()
if __name__ == "__main__":
main()
次の my_executable_table
は my_script.py
の出力から構築されており、my_executable_table
から SELECT
を実行するたびに 10 個のランダムな文字列を生成します:
CREATE TABLE my_executable_table (
x UInt32,
y String
)
ENGINE = Executable('my_script.py', TabSeparated, (SELECT 10))
テーブルを作成しても即座にスクリプトを実行することはなく、my_executable_table
をクエリするとスクリプトが実行されます:
SELECT * FROM my_executable_table
┌─x─┬─y──────────┐
│ 0 │ BsnKBsNGNH │
│ 1 │ mgHfBCUrWM │
│ 2 │ iDQAVhlygr │
│ 3 │ uNGwDuXyCk │
│ 4 │ GcFdQWvoLB │
│ 5 │ UkciuuOTVO │
│ 6 │ HoKeCdHkbs │
│ 7 │ xRvySxqAcR │
│ 8 │ LKbXPHpyDI │
│ 9 │ zxogHTzEVV │
└───┴────────────┘
クエリ結果をスクリプトに渡す
Hacker News のユーザーはコメントを残します。Python には自然言語処理のツールキット(nltk
)があり、コメントがポジティブ、ネガティブ、あるいは中立であるかを判定する SentimentIntensityAnalyzer
が存在します。Hacker News のコメントの感情を nltk
を使用して計算する Executable
テーブルを作成してみましょう。
この例では、このページで説明されている hackernews
テーブルを使用します。hackernews
テーブルには UInt64
タイプの id
カラムと comment
という名前の String
カラムがあります。まずは Executable
テーブルを定義してみましょう:
CREATE TABLE sentiment (
id UInt64,
sentiment Float32
)
ENGINE = Executable(
'sentiment.py',
TabSeparated,
(SELECT id, comment FROM hackernews WHERE id > 0 AND comment != '' LIMIT 20)
);
sentiment
テーブルについていくつかのコメント:
- ファイル
sentiment.py
はuser_scripts
フォルダに保存されています(user_scripts_path
設定のデフォルトフォルダ) TabSeparated
フォーマットは、Python スクリプトがタブ区切りの値を含む生データ行を生成する必要があることを意味します- クエリは
hackernews
から 2 つのカラムを選択しています。Python スクリプトはそれらのカラム値を入力行から解析する必要があります
sentiment.py
の定義はこちらです:
#!/usr/local/bin/python3.9
import sys
import nltk
from nltk.sentiment import SentimentIntensityAnalyzer
def main():
sentiment_analyzer = SentimentIntensityAnalyzer()
while True:
try:
row = sys.stdin.readline()
if row == '':
break
split_line = row.split("\t")
id = str(split_line[0])
comment = split_line[1]
score = sentiment_analyzer.polarity_scores(comment)['compound']
print(id + '\t' + str(score) + '\n', end='')
sys.stdout.flush()
except BaseException as x:
break
if __name__ == "__main__":
main()
Python スクリプトに関するコメント:
- 機能させるためには
nltk.downloader.download('vader_lexicon')
を実行する必要があります。これはスクリプト内に配置することも可能ですが、そうするとsentiment
テーブルの各クエリ実行ごとにダウンロードされるため非効率です row
の各値はSELECT id, comment FROM hackernews WHERE id > 0 AND comment != '' LIMIT 20
の結果セット内の行です- 入力行はタブ区切りなので、Python の
split
関数を使ってid
とcomment
を解析します polarity_scores
の結果は JSON オブジェクトで、いくつかの値を持ちます。ここではその JSON オブジェクトのcompound
値だけを取得することにしました- ClickHouse の
sentiment
テーブルはTabSeparated
フォーマットを使用し、2 つのカラムを含むので、print
関数でそれらのカラムをタブで区切っています
sentiment
テーブルから行を選択するクエリを書くたびに、SELECT id, comment FROM hackernews WHERE id > 0 AND comment != '' LIMIT 20
クエリが実行され、その結果が sentiment.py
に渡されます。試してみましょう:
SELECT *
FROM sentiment
レスポンスは次のようになります:
┌───────id─┬─sentiment─┐
│ 7398199 │ 0.4404 │
│ 21640317 │ 0.1779 │
│ 21462000 │ 0 │
│ 25168863 │ 0 │
│ 25168978 │ -0.1531 │
│ 25169359 │ 0 │
│ 25169394 │ -0.9231 │
│ 25169766 │ 0.4137 │
│ 25172570 │ 0.7469 │
│ 25173687 │ 0.6249 │
│ 28291534 │ 0 │
│ 28291669 │ -0.4767 │
│ 28291731 │ 0 │
│ 28291949 │ -0.4767 │
│ 28292004 │ 0.3612 │
│ 28292050 │ -0.296 │
│ 28292322 │ 0 │
│ 28295172 │ 0.7717 │
│ 28295288 │ 0.4404 │
│ 21465723 │ -0.6956 │
└──────────┴───────────┘
ExecutablePool テーブルの作成
ExecutablePool
の文法は Executable
と似ていますが、ExecutablePool
テーブルにはいくつかの特有の設定があります:
pool_size
- 説明: プロセスプールのサイズ。サイズが0の場合、サイズ制限はありません
- デフォルト値: 16
max_command_execution_time
- 説明: コマンドの最大実行時間(秒単位)
- デフォルト値: 10
上記の sentiment
テーブルを Executable
ではなく ExecutablePool
を使用するように簡単に変換できます:
CREATE TABLE sentiment_pooled (
id UInt64,
sentiment Float32
)
ENGINE = ExecutablePool(
'sentiment.py',
TabSeparated,
(SELECT id, comment FROM hackernews WHERE id > 0 AND comment != '' LIMIT 20000)
)
SETTINGS
pool_size = 4;
クライアントが sentiment_pooled
テーブルをクエリすると、ClickHouse はオンデマンドで 4 つのプロセスを維持します。