chc-r8 0.3.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,137 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'addressable'
4
+ require 'json'
5
+ require 'active_support/deprecation'
6
+ require 'active_support/time'
7
+ require 'active_support/notifications'
8
+ require 'active_support/core_ext/string'
9
+ require 'active_support/core_ext/object/blank'
10
+ require_relative "client/version"
11
+ require_relative "client/database"
12
+ require_relative "client/configuration"
13
+ require_relative "client/bind_index_manager"
14
+ require_relative "client/quoting"
15
+ require_relative "client/arel_engine"
16
+ require_relative "client/query_like"
17
+ require_relative "client/query"
18
+ require_relative "client/redactor"
19
+ require_relative "client/query_builder"
20
+ require_relative "client/formatter"
21
+ require_relative "client/response"
22
+
23
+ module ClickHouse
24
+ module Client
25
+ class << self
26
+ def configuration
27
+ @configuration ||= Configuration.new
28
+ end
29
+
30
+ def configure
31
+ yield(configuration)
32
+ configuration.validate!
33
+ end
34
+ end
35
+
36
+ Error = Class.new(StandardError)
37
+ ConfigurationError = Class.new(Error)
38
+ DatabaseError = Class.new(Error)
39
+ QueryError = Class.new(Error)
40
+
41
+ def self.database_configured?(database, configuration = self.configuration)
42
+ !!configuration.databases[database]
43
+ end
44
+
45
+ # Executes a SELECT database query
46
+ def self.select(query, database, configuration = self.configuration)
47
+ instrumented_execute(query, database, configuration) do |response, instrument|
48
+ parsed_response = configuration.json_parser.parse(response.body)
49
+
50
+ instrument[:statistics] = parsed_response['statistics']&.symbolize_keys
51
+
52
+ Formatter.format(parsed_response)
53
+ end
54
+ end
55
+
56
+ # Executes any kinds of database query without returning any data (INSERT, DELETE)
57
+ def self.execute(query, database, configuration = self.configuration)
58
+ instrumented_execute(query, database, configuration) do |response, instrument|
59
+ expose_summary(response.headers, instrument)
60
+ end
61
+
62
+ true
63
+ end
64
+
65
+ # Inserts a gzip-compressed CSV to ClickHouse
66
+ #
67
+ # Usage:
68
+ #
69
+ # Create a compressed CSV file:
70
+ # > File.binwrite("my_csv.csv", ActiveSupport::Gzip.compress("id\n10\n20"))
71
+ #
72
+ # Invoke the INSERT query:
73
+ # > ClickHouse::Client.insert_csv('INSERT INTO events (id) FORMAT CSV', File.open("my_csv.csv"), :main)
74
+ def self.insert_csv(query, io, database, configuration = self.configuration)
75
+ db = lookup_database(configuration, database)
76
+
77
+ headers = db.headers.merge(
78
+ 'Transfer-Encoding' => 'chunked',
79
+ 'Content-Length' => File.size(io).to_s,
80
+ 'Content-Encoding' => 'gzip'
81
+ )
82
+
83
+ query = ClickHouse::Client::Query.build(query)
84
+ ActiveSupport::Notifications.instrument('sql.click_house', { query:, database: }) do |instrument|
85
+ response = configuration.http_post_proc.call(
86
+ db.build_custom_uri(extra_variables: { query: query.to_sql }).to_s,
87
+ headers,
88
+ io
89
+ )
90
+ raise DatabaseError, response.body unless response.success?
91
+
92
+ expose_summary(response.headers, instrument)
93
+ end
94
+
95
+ true
96
+ end
97
+
98
+ private_class_method def self.expose_summary(headers, instrument)
99
+ return unless headers['x-clickhouse-summary']
100
+
101
+ instrument[:statistics] =
102
+ configuration.json_parser.parse(headers['x-clickhouse-summary']).symbolize_keys
103
+ end
104
+
105
+ private_class_method def self.lookup_database(configuration, database)
106
+ configuration.databases[database].tap do |db|
107
+ raise ConfigurationError, "The database '#{database}' is not configured" unless db
108
+ end
109
+ end
110
+
111
+ private_class_method def self.instrumented_execute(query, database, configuration)
112
+ db = lookup_database(configuration, database)
113
+
114
+ query = ClickHouse::Client::Query.build(query)
115
+
116
+ log_contents = configuration.log_proc.call(query)
117
+ configuration.logger.info(log_contents)
118
+
119
+ ActiveSupport::Notifications.instrument('sql.click_house', { query:, database: }) do |instrument|
120
+ # Use a multipart POST request where the placeholders are sent with the param_ prefix
121
+ # See: https://github.com/ClickHouse/ClickHouse/issues/8842
122
+ query_with_params = query.prepared_placeholders.transform_keys { |key| "param_#{key}" }
123
+ query_with_params['query'] = query.to_sql
124
+
125
+ response = configuration.http_post_proc.call(
126
+ db.uri.to_s,
127
+ db.headers,
128
+ query_with_params
129
+ )
130
+
131
+ raise DatabaseError, response.body unless response.success?
132
+
133
+ yield response, instrument
134
+ end
135
+ end
136
+ end
137
+ end
metadata ADDED
@@ -0,0 +1,192 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: chc-r8
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.3.2
5
+ platform: ruby
6
+ authors:
7
+ - group::optimize
8
+ bindir: bin
9
+ cert_chain: []
10
+ date: 1980-01-02 00:00:00.000000000 Z
11
+ dependencies:
12
+ - !ruby/object:Gem::Dependency
13
+ name: activerecord
14
+ requirement: !ruby/object:Gem::Requirement
15
+ requirements:
16
+ - - "~>"
17
+ - !ruby/object:Gem::Version
18
+ version: '8.0'
19
+ type: :runtime
20
+ prerelease: false
21
+ version_requirements: !ruby/object:Gem::Requirement
22
+ requirements:
23
+ - - "~>"
24
+ - !ruby/object:Gem::Version
25
+ version: '8.0'
26
+ - !ruby/object:Gem::Dependency
27
+ name: activesupport
28
+ requirement: !ruby/object:Gem::Requirement
29
+ requirements:
30
+ - - "~>"
31
+ - !ruby/object:Gem::Version
32
+ version: '8.0'
33
+ type: :runtime
34
+ prerelease: false
35
+ version_requirements: !ruby/object:Gem::Requirement
36
+ requirements:
37
+ - - "~>"
38
+ - !ruby/object:Gem::Version
39
+ version: '8.0'
40
+ - !ruby/object:Gem::Dependency
41
+ name: addressable
42
+ requirement: !ruby/object:Gem::Requirement
43
+ requirements:
44
+ - - "~>"
45
+ - !ruby/object:Gem::Version
46
+ version: '2.8'
47
+ type: :runtime
48
+ prerelease: false
49
+ version_requirements: !ruby/object:Gem::Requirement
50
+ requirements:
51
+ - - "~>"
52
+ - !ruby/object:Gem::Version
53
+ version: '2.8'
54
+ - !ruby/object:Gem::Dependency
55
+ name: json
56
+ requirement: !ruby/object:Gem::Requirement
57
+ requirements:
58
+ - - "~>"
59
+ - !ruby/object:Gem::Version
60
+ version: '2.7'
61
+ type: :runtime
62
+ prerelease: false
63
+ version_requirements: !ruby/object:Gem::Requirement
64
+ requirements:
65
+ - - "~>"
66
+ - !ruby/object:Gem::Version
67
+ version: '2.7'
68
+ - !ruby/object:Gem::Dependency
69
+ name: gitlab-styles
70
+ requirement: !ruby/object:Gem::Requirement
71
+ requirements:
72
+ - - "~>"
73
+ - !ruby/object:Gem::Version
74
+ version: 12.0.1
75
+ type: :development
76
+ prerelease: false
77
+ version_requirements: !ruby/object:Gem::Requirement
78
+ requirements:
79
+ - - "~>"
80
+ - !ruby/object:Gem::Version
81
+ version: 12.0.1
82
+ - !ruby/object:Gem::Dependency
83
+ name: rake
84
+ requirement: !ruby/object:Gem::Requirement
85
+ requirements:
86
+ - - "~>"
87
+ - !ruby/object:Gem::Version
88
+ version: '13.0'
89
+ type: :development
90
+ prerelease: false
91
+ version_requirements: !ruby/object:Gem::Requirement
92
+ requirements:
93
+ - - "~>"
94
+ - !ruby/object:Gem::Version
95
+ version: '13.0'
96
+ - !ruby/object:Gem::Dependency
97
+ name: rspec
98
+ requirement: !ruby/object:Gem::Requirement
99
+ requirements:
100
+ - - "~>"
101
+ - !ruby/object:Gem::Version
102
+ version: '3.0'
103
+ type: :development
104
+ prerelease: false
105
+ version_requirements: !ruby/object:Gem::Requirement
106
+ requirements:
107
+ - - "~>"
108
+ - !ruby/object:Gem::Version
109
+ version: '3.0'
110
+ - !ruby/object:Gem::Dependency
111
+ name: rubocop
112
+ requirement: !ruby/object:Gem::Requirement
113
+ requirements:
114
+ - - ">="
115
+ - !ruby/object:Gem::Version
116
+ version: '0'
117
+ type: :development
118
+ prerelease: false
119
+ version_requirements: !ruby/object:Gem::Requirement
120
+ requirements:
121
+ - - ">="
122
+ - !ruby/object:Gem::Version
123
+ version: '0'
124
+ - !ruby/object:Gem::Dependency
125
+ name: rubocop-rspec
126
+ requirement: !ruby/object:Gem::Requirement
127
+ requirements:
128
+ - - ">="
129
+ - !ruby/object:Gem::Version
130
+ version: '0'
131
+ type: :development
132
+ prerelease: false
133
+ version_requirements: !ruby/object:Gem::Requirement
134
+ requirements:
135
+ - - ">="
136
+ - !ruby/object:Gem::Version
137
+ version: '0'
138
+ description: This Gem provides a simple way to query ClickHouse databases using the
139
+ HTTP interface.
140
+ email:
141
+ - engineering@gitlab.com
142
+ executables: []
143
+ extensions: []
144
+ extra_rdoc_files: []
145
+ files:
146
+ - ".github/workflows/publish.yml"
147
+ - ".gitlab-ci.yml"
148
+ - ".rspec"
149
+ - ".rubocop.yml"
150
+ - ".ruby-version"
151
+ - CODE_OF_CONDUCT.md
152
+ - CONTRIBUTING.md
153
+ - Gemfile
154
+ - Gemfile.lock
155
+ - LICENSE.txt
156
+ - README.md
157
+ - click_house-client.gemspec
158
+ - lib/click_house/client.rb
159
+ - lib/click_house/client/arel_engine.rb
160
+ - lib/click_house/client/bind_index_manager.rb
161
+ - lib/click_house/client/configuration.rb
162
+ - lib/click_house/client/database.rb
163
+ - lib/click_house/client/formatter.rb
164
+ - lib/click_house/client/query.rb
165
+ - lib/click_house/client/query_builder.rb
166
+ - lib/click_house/client/query_like.rb
167
+ - lib/click_house/client/quoting.rb
168
+ - lib/click_house/client/redactor.rb
169
+ - lib/click_house/client/response.rb
170
+ - lib/click_house/client/version.rb
171
+ homepage: https://gitlab.com/gitlab-org/ruby/gems/clickhouse-client
172
+ licenses:
173
+ - MIT
174
+ metadata: {}
175
+ rdoc_options: []
176
+ require_paths:
177
+ - lib
178
+ required_ruby_version: !ruby/object:Gem::Requirement
179
+ requirements:
180
+ - - ">="
181
+ - !ruby/object:Gem::Version
182
+ version: '3.1'
183
+ required_rubygems_version: !ruby/object:Gem::Requirement
184
+ requirements:
185
+ - - ">="
186
+ - !ruby/object:Gem::Version
187
+ version: '0'
188
+ requirements: []
189
+ rubygems_version: 3.7.1
190
+ specification_version: 4
191
+ summary: GitLab's client to interact with ClickHouse
192
+ test_files: []