google-cloud-spanner 2.24.0 → 2.26.0
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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +16 -0
- data/CONTRIBUTING.md +1 -1
- data/lib/google/cloud/spanner/admin/database.rb +21 -9
- data/lib/google/cloud/spanner/admin/instance.rb +21 -9
- data/lib/google/cloud/spanner/backup/job/list.rb +1 -1
- data/lib/google/cloud/spanner/client.rb +4 -1
- data/lib/google/cloud/spanner/commit.rb +44 -52
- data/lib/google/cloud/spanner/convert.rb +30 -3
- data/lib/google/cloud/spanner/database/job/list.rb +1 -1
- data/lib/google/cloud/spanner/database.rb +40 -10
- data/lib/google/cloud/spanner/fields.rb +18 -9
- data/lib/google/cloud/spanner/partition.rb +1 -1
- data/lib/google/cloud/spanner/pool.rb +2 -2
- data/lib/google/cloud/spanner/project.rb +9 -0
- data/lib/google/cloud/spanner/service.rb +35 -7
- data/lib/google/cloud/spanner/session.rb +1 -0
- data/lib/google/cloud/spanner/version.rb +1 -1
- data/lib/google/cloud/spanner.rb +4 -2
- data/lib/google-cloud-spanner.rb +3 -6
- metadata +18 -39
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: cafab317421fcff06f127fb2d156c722f67b8e5d1914ae68d3e0ec8b56db4f76
|
4
|
+
data.tar.gz: 26820cba11c9d0a94d5dbaaff4c28ea0581c25ef3d7a9876d39ef5ca42fe5edc
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 58624fe569f909d5923f897fc02c6865c4861dd289ec5b1f58aead9ae01b87968ebd7b5e23d0305e4661011fd4bc2e3479cc08c08ef0c19b320daa2e68300fca
|
7
|
+
data.tar.gz: c011acec4ad559505de0b89e77dae6d2046f7e00768a421f4bd347ff25a3b8ee35673f1714037528e1d13572e054486ded0fd666d253df9b322a2c90ef19a1bf
|
data/CHANGELOG.md
CHANGED
@@ -1,5 +1,21 @@
|
|
1
1
|
# Release History
|
2
2
|
|
3
|
+
### 2.26.0 (2025-03-24)
|
4
|
+
|
5
|
+
#### Features
|
6
|
+
|
7
|
+
* Support setting the universe domain ([#144](https://github.com/googleapis/ruby-spanner/issues/144))
|
8
|
+
#### Bug Fixes
|
9
|
+
|
10
|
+
* Corrected algorithm that releases stale sessions in the pool ([#156](https://github.com/googleapis/ruby-spanner/issues/156))
|
11
|
+
|
12
|
+
### 2.25.0 (2025-01-29)
|
13
|
+
|
14
|
+
#### Features
|
15
|
+
|
16
|
+
* Support Protobuf Columns ([#124](https://github.com/googleapis/ruby-spanner/issues/124))
|
17
|
+
* Update minimum Ruby version to 3.0 ([#140](https://github.com/googleapis/ruby-spanner/issues/140))
|
18
|
+
|
3
19
|
### 2.24.0 (2024-08-27)
|
4
20
|
|
5
21
|
#### Features
|
data/CONTRIBUTING.md
CHANGED
@@ -24,7 +24,7 @@ be able to accept your pull requests.
|
|
24
24
|
In order to use the google-cloud-spanner console and run the project's tests,
|
25
25
|
there is a small amount of setup:
|
26
26
|
|
27
|
-
1. Install Ruby. google-cloud-spanner requires Ruby
|
27
|
+
1. Install Ruby. google-cloud-spanner requires Ruby 3.0+. You may choose to
|
28
28
|
manage your Ruby and gem installations with [RVM](https://rvm.io/),
|
29
29
|
[rbenv](https://github.com/rbenv/rbenv), or
|
30
30
|
[chruby](https://github.com/postmodern/chruby).
|
@@ -85,6 +85,7 @@ module Google
|
|
85
85
|
#
|
86
86
|
def self.database_admin project_id: nil,
|
87
87
|
credentials: nil,
|
88
|
+
universe_domain: nil,
|
88
89
|
scope: nil,
|
89
90
|
timeout: nil,
|
90
91
|
endpoint: nil,
|
@@ -93,14 +94,23 @@ module Google
|
|
93
94
|
emulator_host: nil,
|
94
95
|
lib_name: nil,
|
95
96
|
lib_version: nil
|
96
|
-
project_id
|
97
|
-
scope
|
98
|
-
timeout
|
97
|
+
project_id ||= project || default_project_id
|
98
|
+
scope ||= configure.scope
|
99
|
+
timeout ||= configure.timeout
|
99
100
|
emulator_host ||= configure.emulator_host
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
101
|
+
# TODO: This logic is part of UniverseDomainConcerns in gapic-common
|
102
|
+
# but is being copied here because we need to determine the host up
|
103
|
+
# front in order to build a gRPC channel. We should refactor this
|
104
|
+
# somehow to allow this logic to live where it is supposed to.
|
105
|
+
universe_domain ||= configure.universe_domain || ENV["GOOGLE_CLOUD_UNIVERSE_DOMAIN"] || "googleapis.com"
|
106
|
+
endpoint ||= emulator_host || configure.endpoint
|
107
|
+
endpoint ||=
|
108
|
+
Google::Cloud::Spanner::Admin::Database::V1::DatabaseAdmin::Client::DEFAULT_ENDPOINT_TEMPLATE.sub(
|
109
|
+
Gapic::UniverseDomainConcerns::ENDPOINT_SUBSTITUTION, universe_domain
|
110
|
+
)
|
111
|
+
credentials ||= keyfile
|
112
|
+
lib_name ||= configure.lib_name
|
113
|
+
lib_version ||= configure.lib_version
|
104
114
|
|
105
115
|
if emulator_host
|
106
116
|
credentials = :this_channel_is_insecure
|
@@ -121,10 +131,11 @@ module Google
|
|
121
131
|
configure.quota_project ||= credentials.quota_project_id if credentials.respond_to? :quota_project_id
|
122
132
|
|
123
133
|
Admin::Database::V1::DatabaseAdmin::Client.new do |config|
|
134
|
+
config.universe_domain = universe_domain
|
124
135
|
config.credentials = channel endpoint, credentials
|
125
136
|
config.quota_project = configure.quota_project
|
126
137
|
config.timeout = timeout if timeout
|
127
|
-
config.endpoint = endpoint
|
138
|
+
config.endpoint = endpoint
|
128
139
|
config.lib_name = lib_name_with_prefix lib_name, lib_version
|
129
140
|
config.lib_version = Google::Cloud::Spanner::VERSION
|
130
141
|
config.metadata = { "google-cloud-resource-prefix" => "projects/#{project_id}" }
|
@@ -290,7 +301,7 @@ module Google
|
|
290
301
|
class Configuration
|
291
302
|
extend ::Gapic::Config
|
292
303
|
|
293
|
-
config_attr :endpoint,
|
304
|
+
config_attr :endpoint, nil, ::String
|
294
305
|
config_attr :credentials, nil do |value|
|
295
306
|
allowed = [::String, ::Hash, ::Google::Auth::Credentials, ::Signet::OAuth2::Client, nil]
|
296
307
|
allowed += [::GRPC::Core::Channel, ::GRPC::Core::ChannelCredentials] if defined? ::GRPC
|
@@ -307,6 +318,7 @@ module Google
|
|
307
318
|
config_attr :query_options, nil, ::Hash, nil
|
308
319
|
config_attr :metadata, nil, ::Hash, nil
|
309
320
|
config_attr :retry_policy, nil, ::Hash, nil
|
321
|
+
config_attr :universe_domain, nil, ::String, nil
|
310
322
|
|
311
323
|
# @private
|
312
324
|
def initialize parent_config = nil
|
@@ -85,6 +85,7 @@ module Google
|
|
85
85
|
#
|
86
86
|
def self.instance_admin project_id: nil,
|
87
87
|
credentials: nil,
|
88
|
+
universe_domain: nil,
|
88
89
|
scope: nil,
|
89
90
|
timeout: nil,
|
90
91
|
endpoint: nil,
|
@@ -93,14 +94,23 @@ module Google
|
|
93
94
|
emulator_host: nil,
|
94
95
|
lib_name: nil,
|
95
96
|
lib_version: nil
|
96
|
-
project_id
|
97
|
-
scope
|
98
|
-
timeout
|
97
|
+
project_id ||= project || default_project_id
|
98
|
+
scope ||= configure.scope
|
99
|
+
timeout ||= configure.timeout
|
99
100
|
emulator_host ||= configure.emulator_host
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
101
|
+
# TODO: This logic is part of UniverseDomainConcerns in gapic-common
|
102
|
+
# but is being copied here because we need to determine the host up
|
103
|
+
# front in order to build a gRPC channel. We should refactor this
|
104
|
+
# somehow to allow this logic to live where it is supposed to.
|
105
|
+
universe_domain ||= configure.universe_domain || ENV["GOOGLE_CLOUD_UNIVERSE_DOMAIN"] || "googleapis.com"
|
106
|
+
endpoint ||= emulator_host || configure.endpoint
|
107
|
+
endpoint ||=
|
108
|
+
Google::Cloud::Spanner::Admin::Instance::V1::InstanceAdmin::Client::DEFAULT_ENDPOINT_TEMPLATE.sub(
|
109
|
+
Gapic::UniverseDomainConcerns::ENDPOINT_SUBSTITUTION, universe_domain
|
110
|
+
)
|
111
|
+
credentials ||= keyfile
|
112
|
+
lib_name ||= configure.lib_name
|
113
|
+
lib_version ||= configure.lib_version
|
104
114
|
|
105
115
|
if emulator_host
|
106
116
|
credentials = :this_channel_is_insecure
|
@@ -121,10 +131,11 @@ module Google
|
|
121
131
|
configure.quota_project ||= credentials.quota_project_id if credentials.respond_to? :quota_project_id
|
122
132
|
|
123
133
|
Admin::Instance::V1::InstanceAdmin::Client.new do |config|
|
134
|
+
config.universe_domain = universe_domain
|
124
135
|
config.credentials = channel endpoint, credentials
|
125
136
|
config.quota_project = configure.quota_project
|
126
137
|
config.timeout = timeout if timeout
|
127
|
-
config.endpoint = endpoint
|
138
|
+
config.endpoint = endpoint
|
128
139
|
config.lib_name = lib_name_with_prefix lib_name, lib_version
|
129
140
|
config.lib_version = Google::Cloud::Spanner::VERSION
|
130
141
|
config.metadata = { "google-cloud-resource-prefix" => "projects/#{project_id}" }
|
@@ -290,7 +301,7 @@ module Google
|
|
290
301
|
class Configuration
|
291
302
|
extend ::Gapic::Config
|
292
303
|
|
293
|
-
config_attr :endpoint,
|
304
|
+
config_attr :endpoint, nil, ::String
|
294
305
|
config_attr :credentials, nil do |value|
|
295
306
|
allowed = [::String, ::Hash, ::Google::Auth::Credentials, ::Signet::OAuth2::Client, nil]
|
296
307
|
allowed += [::GRPC::Core::Channel, ::GRPC::Core::ChannelCredentials] if defined? ::GRPC
|
@@ -307,6 +318,7 @@ module Google
|
|
307
318
|
config_attr :query_options, nil, ::Hash, nil
|
308
319
|
config_attr :metadata, nil, ::Hash, nil
|
309
320
|
config_attr :retry_policy, nil, ::Hash, nil
|
321
|
+
config_attr :universe_domain, nil, ::String, nil
|
310
322
|
|
311
323
|
# @private
|
312
324
|
def initialize parent_config = nil
|
@@ -156,7 +156,7 @@ module Google
|
|
156
156
|
# object. Operation object is a backup operation.
|
157
157
|
#
|
158
158
|
def self.from_grpc grpc, service
|
159
|
-
operations_client =
|
159
|
+
operations_client =
|
160
160
|
service.databases.instance_variable_get "@operations_client"
|
161
161
|
jobs = new(Array(grpc.response.operations).map do |job_grpc|
|
162
162
|
Job.from_grpc \
|
@@ -1022,10 +1022,10 @@ module Google
|
|
1022
1022
|
# | `TIMESTAMP` | `Time`, `DateTime` | |
|
1023
1023
|
# | `BYTES` | `File`, `IO`, `StringIO`, or similar | |
|
1024
1024
|
# | `ARRAY` | `Array` | Nested arrays are not supported. |
|
1025
|
+
# | `PROTO` | Determined by proto_fqn | |
|
1025
1026
|
#
|
1026
1027
|
# See [Data
|
1027
1028
|
# types](https://cloud.google.com/spanner/docs/data-definition-language#data_types).
|
1028
|
-
#
|
1029
1029
|
# @param [Boolean] exclude_txn_from_change_streams If set to true,
|
1030
1030
|
# mutations will not be recorded in change streams with DDL option
|
1031
1031
|
# `allow_txn_exclusion=true`.
|
@@ -1169,6 +1169,7 @@ module Google
|
|
1169
1169
|
# | `TIMESTAMP` | `Time`, `DateTime` | |
|
1170
1170
|
# | `BYTES` | `File`, `IO`, `StringIO`, or similar | |
|
1171
1171
|
# | `ARRAY` | `Array` | Nested arrays are not supported. |
|
1172
|
+
# | `PROTO` | Determined by proto_fqn | |
|
1172
1173
|
#
|
1173
1174
|
# See [Data
|
1174
1175
|
# types](https://cloud.google.com/spanner/docs/data-definition-language#data_types).
|
@@ -1314,6 +1315,7 @@ module Google
|
|
1314
1315
|
# | `TIMESTAMP` | `Time`, `DateTime` | |
|
1315
1316
|
# | `BYTES` | `File`, `IO`, `StringIO`, or similar | |
|
1316
1317
|
# | `ARRAY` | `Array` | Nested arrays are not supported. |
|
1318
|
+
# | `PROTO` | Determined by proto_fqn | |
|
1317
1319
|
#
|
1318
1320
|
# See [Data
|
1319
1321
|
# types](https://cloud.google.com/spanner/docs/data-definition-language#data_types).
|
@@ -1460,6 +1462,7 @@ module Google
|
|
1460
1462
|
# | `TIMESTAMP` | `Time`, `DateTime` | |
|
1461
1463
|
# | `BYTES` | `File`, `IO`, `StringIO`, or similar | |
|
1462
1464
|
# | `ARRAY` | `Array` | Nested arrays are not supported. |
|
1465
|
+
# | `PROTO` | Determined by proto_fqn | |
|
1463
1466
|
#
|
1464
1467
|
# See [Data
|
1465
1468
|
# types](https://cloud.google.com/spanner/docs/data-definition-language#data_types).
|
@@ -76,6 +76,7 @@ module Google
|
|
76
76
|
# | `TIMESTAMP` | `Time`, `DateTime` | |
|
77
77
|
# | `BYTES` | `File`, `IO`, `StringIO`, or similar | |
|
78
78
|
# | `ARRAY` | `Array` | Nested arrays are not supported. |
|
79
|
+
# | `PROTO` | Determined by proto_fqn | |
|
79
80
|
#
|
80
81
|
# See [Data
|
81
82
|
# types](https://cloud.google.com/spanner/docs/data-definition-language#data_types).
|
@@ -93,19 +94,7 @@ module Google
|
|
93
94
|
# end
|
94
95
|
#
|
95
96
|
def upsert table, *rows
|
96
|
-
|
97
|
-
return rows if rows.empty?
|
98
|
-
rows.compact
|
99
|
-
rows.delete_if(&:empty?)
|
100
|
-
@mutations += rows.map do |row|
|
101
|
-
V1::Mutation.new(
|
102
|
-
insert_or_update: V1::Mutation::Write.new(
|
103
|
-
table: table, columns: row.keys.map(&:to_s),
|
104
|
-
values: [Convert.object_to_grpc_value(row.values).list_value]
|
105
|
-
)
|
106
|
-
)
|
107
|
-
end
|
108
|
-
rows
|
97
|
+
mutations_from_rows table, rows, "insert_or_update"
|
109
98
|
end
|
110
99
|
alias save upsert
|
111
100
|
|
@@ -136,6 +125,7 @@ module Google
|
|
136
125
|
# | `TIMESTAMP` | `Time`, `DateTime` | |
|
137
126
|
# | `BYTES` | `File`, `IO`, `StringIO`, or similar | |
|
138
127
|
# | `ARRAY` | `Array` | Nested arrays are not supported. |
|
128
|
+
# | `PROTO` | Determined by proto_fqn | |
|
139
129
|
#
|
140
130
|
# See [Data
|
141
131
|
# types](https://cloud.google.com/spanner/docs/data-definition-language#data_types).
|
@@ -153,19 +143,7 @@ module Google
|
|
153
143
|
# end
|
154
144
|
#
|
155
145
|
def insert table, *rows
|
156
|
-
|
157
|
-
return rows if rows.empty?
|
158
|
-
rows.compact
|
159
|
-
rows.delete_if(&:empty?)
|
160
|
-
@mutations += rows.map do |row|
|
161
|
-
V1::Mutation.new(
|
162
|
-
insert: V1::Mutation::Write.new(
|
163
|
-
table: table, columns: row.keys.map(&:to_s),
|
164
|
-
values: [Convert.object_to_grpc_value(row.values).list_value]
|
165
|
-
)
|
166
|
-
)
|
167
|
-
end
|
168
|
-
rows
|
146
|
+
mutations_from_rows table, rows, "insert"
|
169
147
|
end
|
170
148
|
|
171
149
|
##
|
@@ -195,6 +173,7 @@ module Google
|
|
195
173
|
# | `TIMESTAMP` | `Time`, `DateTime` | |
|
196
174
|
# | `BYTES` | `File`, `IO`, `StringIO`, or similar | |
|
197
175
|
# | `ARRAY` | `Array` | Nested arrays are not supported. |
|
176
|
+
# | `PROTO` | Determined by proto_fqn | |
|
198
177
|
#
|
199
178
|
# See [Data
|
200
179
|
# types](https://cloud.google.com/spanner/docs/data-definition-language#data_types).
|
@@ -212,19 +191,7 @@ module Google
|
|
212
191
|
# end
|
213
192
|
#
|
214
193
|
def update table, *rows
|
215
|
-
|
216
|
-
return rows if rows.empty?
|
217
|
-
rows.compact
|
218
|
-
rows.delete_if(&:empty?)
|
219
|
-
@mutations += rows.map do |row|
|
220
|
-
V1::Mutation.new(
|
221
|
-
update: V1::Mutation::Write.new(
|
222
|
-
table: table, columns: row.keys.map(&:to_s),
|
223
|
-
values: [Convert.object_to_grpc_value(row.values).list_value]
|
224
|
-
)
|
225
|
-
)
|
226
|
-
end
|
227
|
-
rows
|
194
|
+
mutations_from_rows table, rows, "update"
|
228
195
|
end
|
229
196
|
|
230
197
|
##
|
@@ -256,6 +223,7 @@ module Google
|
|
256
223
|
# | `TIMESTAMP` | `Time`, `DateTime` | |
|
257
224
|
# | `BYTES` | `File`, `IO`, `StringIO`, or similar | |
|
258
225
|
# | `ARRAY` | `Array` | Nested arrays are not supported. |
|
226
|
+
# | `PROTO` | Determined by proto_fqn | |
|
259
227
|
#
|
260
228
|
# See [Data
|
261
229
|
# types](https://cloud.google.com/spanner/docs/data-definition-language#data_types).
|
@@ -273,19 +241,7 @@ module Google
|
|
273
241
|
# end
|
274
242
|
#
|
275
243
|
def replace table, *rows
|
276
|
-
|
277
|
-
return rows if rows.empty?
|
278
|
-
rows.compact
|
279
|
-
rows.delete_if(&:empty?)
|
280
|
-
@mutations += rows.map do |row|
|
281
|
-
V1::Mutation.new(
|
282
|
-
replace: V1::Mutation::Write.new(
|
283
|
-
table: table, columns: row.keys.map(&:to_s),
|
284
|
-
values: [Convert.object_to_grpc_value(row.values).list_value]
|
285
|
-
)
|
286
|
-
)
|
287
|
-
end
|
288
|
-
rows
|
244
|
+
mutations_from_rows table, rows, "replace"
|
289
245
|
end
|
290
246
|
|
291
247
|
##
|
@@ -330,6 +286,42 @@ module Google
|
|
330
286
|
|
331
287
|
protected
|
332
288
|
|
289
|
+
##
|
290
|
+
# @private
|
291
|
+
# Generates mutations from `rows` to be performed on a given table, converting
|
292
|
+
# given rows to their corresponding column and gRPC values.
|
293
|
+
#
|
294
|
+
# @param [String] table The name of the table in the database to be
|
295
|
+
# modified.
|
296
|
+
# @param [Array<Hash>] rows One or more hash objects with the hash keys
|
297
|
+
# matching the table's columns, and the hash values matching the
|
298
|
+
# table's values.
|
299
|
+
# @param [String] type The type of mutation to be performed.
|
300
|
+
#
|
301
|
+
def mutations_from_rows table, rows, type
|
302
|
+
rows = Array(rows).flatten
|
303
|
+
return rows if rows.empty?
|
304
|
+
rows.compact
|
305
|
+
rows.delete_if { |row| row.respond_to?(:empty?) && row.empty? }
|
306
|
+
@mutations += rows.map do |row|
|
307
|
+
# This case applies whenever a Protobuf object is the row itself, and not part of individual column fields.
|
308
|
+
if row.class.respond_to? :descriptor
|
309
|
+
columns = row.class.descriptor.map(&:name)
|
310
|
+
values = [Google::Protobuf::ListValue.new(values: [Convert.object_to_grpc_value(row, :PROTO)])]
|
311
|
+
else
|
312
|
+
columns = row.keys.map(&:to_s)
|
313
|
+
values = [Convert.object_to_grpc_value(row.values).list_value]
|
314
|
+
end
|
315
|
+
V1::Mutation.new(
|
316
|
+
"#{type}": V1::Mutation::Write.new(
|
317
|
+
table: table, columns: columns,
|
318
|
+
values: values
|
319
|
+
)
|
320
|
+
)
|
321
|
+
end
|
322
|
+
rows
|
323
|
+
end
|
324
|
+
|
333
325
|
def key_set keys
|
334
326
|
return V1::KeySet.new all: true if keys.nil?
|
335
327
|
keys = [keys] unless keys.is_a? Array
|
@@ -45,9 +45,16 @@ module Google
|
|
45
45
|
end
|
46
46
|
|
47
47
|
field ||= field_for_object obj
|
48
|
-
[object_to_grpc_value(obj, field), grpc_type_for_field(field)]
|
48
|
+
[object_to_grpc_value(obj, field), grpc_type_for_field(field, obj)]
|
49
49
|
end
|
50
50
|
|
51
|
+
##
|
52
|
+
# @private
|
53
|
+
# Convert objects to their corresponding gRPC values.
|
54
|
+
#
|
55
|
+
# `field` is used to determine whether the object itself is a value or
|
56
|
+
# a collection of values.
|
57
|
+
#
|
51
58
|
def object_to_grpc_value obj, field = nil
|
52
59
|
obj = obj.to_column_value if obj.respond_to? :to_column_value
|
53
60
|
|
@@ -101,12 +108,17 @@ module Google
|
|
101
108
|
else
|
102
109
|
Google::Protobuf::Value.new string_value: obj.to_json
|
103
110
|
end
|
111
|
+
when Google::Protobuf::MessageExts
|
112
|
+
proto_class = obj.class
|
113
|
+
content = proto_class.encode obj
|
114
|
+
encoded_content = Base64.strict_encode64(content)
|
115
|
+
Google::Protobuf::Value.new string_value: encoded_content
|
104
116
|
else
|
105
117
|
if obj.respond_to?(:read) && obj.respond_to?(:rewind)
|
106
118
|
obj.rewind
|
107
119
|
content = obj.read.force_encoding("ASCII-8BIT")
|
108
120
|
encoded_content = Base64.strict_encode64(content)
|
109
|
-
Google::Protobuf::Value.new
|
121
|
+
Google::Protobuf::Value.new string_value: encoded_content
|
110
122
|
else
|
111
123
|
raise ArgumentError,
|
112
124
|
"A value of type #{obj.class} is not supported."
|
@@ -161,6 +173,8 @@ module Google
|
|
161
173
|
Fields.new Hash[raw_type_pairs]
|
162
174
|
when Data
|
163
175
|
obj.fields
|
176
|
+
when Google::Protobuf::MessageExts
|
177
|
+
:PROTO
|
164
178
|
else
|
165
179
|
if obj.respond_to?(:read) && obj.respond_to?(:rewind)
|
166
180
|
:BYTES
|
@@ -171,7 +185,7 @@ module Google
|
|
171
185
|
end
|
172
186
|
end
|
173
187
|
|
174
|
-
def grpc_type_for_field field
|
188
|
+
def grpc_type_for_field field, obj = nil
|
175
189
|
return field.to_grpc_type if field.respond_to? :to_grpc_type
|
176
190
|
|
177
191
|
case field
|
@@ -184,11 +198,20 @@ module Google
|
|
184
198
|
V1::Type.new(code: :NUMERIC, type_annotation: :PG_NUMERIC)
|
185
199
|
when :PG_JSONB
|
186
200
|
V1::Type.new(code: :JSON, type_annotation: :PG_JSONB)
|
201
|
+
when :PROTO
|
202
|
+
V1::Type.new(code: :PROTO, proto_type_fqn: obj.nil? ? "" : obj.class.descriptor.name)
|
187
203
|
else
|
188
204
|
V1::Type.new(code: field)
|
189
205
|
end
|
190
206
|
end
|
191
207
|
|
208
|
+
##
|
209
|
+
# Converts a gRPC value to a Ruby object.
|
210
|
+
#
|
211
|
+
# @param [Google::Protobuf::Value] value The gRPC value to convert.
|
212
|
+
# @param [Google::Spanner::V1::Type] type The underlying type for data.
|
213
|
+
# @return [::Object] The Ruby object that represents the value, converted to the closest
|
214
|
+
# matching Ruby class based on the gRPC type.
|
192
215
|
def grpc_value_to_object value, type
|
193
216
|
return nil if value.kind == :null_value
|
194
217
|
|
@@ -229,6 +252,10 @@ module Google
|
|
229
252
|
BigDecimal value.string_value
|
230
253
|
when :JSON
|
231
254
|
JSON.parse value.string_value
|
255
|
+
when :PROTO
|
256
|
+
descriptor = Google::Protobuf::DescriptorPool.generated_pool.lookup(type.proto_type_fqn).msgclass
|
257
|
+
content = Base64.decode64 value.string_value
|
258
|
+
descriptor.decode content
|
232
259
|
end
|
233
260
|
end
|
234
261
|
|
@@ -156,7 +156,7 @@ module Google
|
|
156
156
|
# object. Operation object is a database operation.
|
157
157
|
#
|
158
158
|
def self.from_grpc grpc, service
|
159
|
-
operations_client =
|
159
|
+
operations_client =
|
160
160
|
service.databases.instance_variable_get "@operations_client"
|
161
161
|
jobs = new(Array(grpc.response.operations).map do |job_grpc|
|
162
162
|
Job.from_grpc \
|
@@ -238,6 +238,10 @@ module Google
|
|
238
238
|
# valid identifier: `[a-z][a-z0-9_]*`. Will raise
|
239
239
|
# {Google::Cloud::AlreadyExistsError} if the named operation already
|
240
240
|
# exists. Optional.
|
241
|
+
# @param [Google::Protobuf::FileDescriptorSet, String] descriptor_set The file
|
242
|
+
# descriptor set object to be used in the update, or alternatively, an absolute
|
243
|
+
# path to the generated file descriptor set. The descriptor set is only used
|
244
|
+
# during DDL statements, such as `CREATE PROTO BUNDLE`.
|
241
245
|
#
|
242
246
|
# @return [Database::Job] The job representing the long-running,
|
243
247
|
# asynchronous processing of a database schema update operation.
|
@@ -248,22 +252,48 @@ module Google
|
|
248
252
|
# spanner = Google::Cloud::Spanner.new
|
249
253
|
# database = spanner.database "my-instance", "my-database"
|
250
254
|
#
|
251
|
-
# add_users_table_sql =
|
252
|
-
#
|
253
|
-
#
|
254
|
-
#
|
255
|
-
#
|
256
|
-
#
|
257
|
-
#
|
258
|
-
#
|
255
|
+
# add_users_table_sql =
|
256
|
+
# <<~SQL
|
257
|
+
# CREATE TABLE users (
|
258
|
+
# id INT64 NOT NULL,
|
259
|
+
# username STRING(25) NOT NULL,
|
260
|
+
# name STRING(45) NOT NULL,
|
261
|
+
# email STRING(128),
|
262
|
+
# ) PRIMARY KEY(id)
|
263
|
+
# SQL
|
259
264
|
#
|
260
265
|
# database.update statements: [add_users_table_sql]
|
261
266
|
#
|
262
|
-
|
267
|
+
# @example
|
268
|
+
# require "google/cloud/spanner"
|
269
|
+
#
|
270
|
+
# spanner = Google::Cloud::Spanner.new
|
271
|
+
# database = spanner.database "my-instance", "my-database"
|
272
|
+
#
|
273
|
+
# create_proto_bundle_sql =
|
274
|
+
# <<~SQL
|
275
|
+
# CREATE PROTO BUNDLE (
|
276
|
+
# `examples.User`
|
277
|
+
# )
|
278
|
+
# SQL
|
279
|
+
#
|
280
|
+
# create_users_table_sql =
|
281
|
+
# <<~SQL
|
282
|
+
# CREATE TABLE users (
|
283
|
+
# id INT64 NOT NULL,
|
284
|
+
# user `examples.User` NOT NULL
|
285
|
+
# ) PRIMARY KEY (id)
|
286
|
+
# SQL
|
287
|
+
#
|
288
|
+
# database.update statements: [create_proto_bundle_sql, create_users_table_sql],
|
289
|
+
# descriptor_set: "/usr/local/user_descriptors.pb"
|
290
|
+
#
|
291
|
+
def update statements: [], operation_id: nil, descriptor_set: nil
|
263
292
|
ensure_service!
|
264
293
|
grpc = service.update_database_ddl instance_id, database_id,
|
265
294
|
statements: statements,
|
266
|
-
operation_id: operation_id
|
295
|
+
operation_id: operation_id,
|
296
|
+
descriptor_set: descriptor_set
|
267
297
|
Database::Job.from_grpc grpc, service
|
268
298
|
end
|
269
299
|
|
@@ -68,6 +68,7 @@ module Google
|
|
68
68
|
# * `:INT64`
|
69
69
|
# * `:STRING`
|
70
70
|
# * `:TIMESTAMP`
|
71
|
+
# * `:PROTO`
|
71
72
|
# * `Array` - Lists are specified by providing the type code in an
|
72
73
|
# array. For example, an array of integers are specified as
|
73
74
|
# `[:INT64]`.
|
@@ -275,24 +276,23 @@ module Google
|
|
275
276
|
return Data.from_grpc nil, @grpc_fields
|
276
277
|
elsif data.is_a? Array
|
277
278
|
# Convert data in the order it was recieved
|
278
|
-
|
279
|
-
Convert.object_to_grpc_value_and_type(datum, cached_types[index])
|
279
|
+
values_and_types = data.map.with_index do |datum, index|
|
280
|
+
Convert.object_to_grpc_value_and_type(datum, cached_types[index])
|
280
281
|
end
|
281
|
-
return Data.from_grpc values, @grpc_fields
|
282
282
|
elsif data.is_a? Hash
|
283
283
|
# Pull values from hash in order of the fields,
|
284
284
|
# we can't always trust the Hash to be in order.
|
285
|
-
|
285
|
+
values_and_types = @grpc_fields.map.with_index do |field, index|
|
286
286
|
if data.key? index
|
287
287
|
Convert.object_to_grpc_value_and_type(data[index],
|
288
|
-
cached_types[index])
|
288
|
+
cached_types[index])
|
289
289
|
elsif !field.name.to_s.empty?
|
290
290
|
if data.key? field.name.to_s
|
291
291
|
Convert.object_to_grpc_value_and_type(data[field.name.to_s],
|
292
|
-
cached_types[index])
|
292
|
+
cached_types[index])
|
293
293
|
elsif data.key? field.name.to_s.to_sym
|
294
294
|
Convert.object_to_grpc_value_and_type(data[field.name.to_s.to_sym],
|
295
|
-
cached_types[index])
|
295
|
+
cached_types[index])
|
296
296
|
else
|
297
297
|
raise "data value for field #{field.name} missing"
|
298
298
|
end
|
@@ -300,9 +300,18 @@ module Google
|
|
300
300
|
raise "data value for field #{index} missing"
|
301
301
|
end
|
302
302
|
end
|
303
|
-
|
303
|
+
else
|
304
|
+
raise ArgumentError, "can only accept Array or Hash"
|
305
|
+
end
|
306
|
+
|
307
|
+
# This is not ideal since we loop through `@grpc_fields` a second time after
|
308
|
+
# initialization. Refactoring can be done to perform this step later on when
|
309
|
+
# all information for the type is available.
|
310
|
+
values, grpc_types = values_and_types.transpose
|
311
|
+
grpc_types&.each_with_index do |grpc_type, index|
|
312
|
+
@grpc_fields[index].type = grpc_type
|
304
313
|
end
|
305
|
-
|
314
|
+
Data.from_grpc values, @grpc_fields
|
306
315
|
end
|
307
316
|
alias data struct
|
308
317
|
alias new struct
|
@@ -136,10 +136,10 @@ module Google
|
|
136
136
|
|
137
137
|
@mutex.synchronize do
|
138
138
|
available_count = sessions_available.count
|
139
|
-
release_count =
|
139
|
+
release_count = available_count - @min
|
140
140
|
release_count = 0 if release_count.negative?
|
141
141
|
|
142
|
-
to_keepalive
|
142
|
+
to_keepalive = sessions_available.select do |x|
|
143
143
|
x.idle_since? @keepalive
|
144
144
|
end
|
145
145
|
|
@@ -95,6 +95,15 @@ module Google
|
|
95
95
|
end
|
96
96
|
alias project project_id
|
97
97
|
|
98
|
+
##
|
99
|
+
# The universe domain the client is connected to
|
100
|
+
#
|
101
|
+
# @return [String]
|
102
|
+
#
|
103
|
+
def universe_domain
|
104
|
+
service.universe_domain
|
105
|
+
end
|
106
|
+
|
98
107
|
##
|
99
108
|
# Retrieves the list of Cloud Spanner instances for the project.
|
100
109
|
#
|
@@ -38,6 +38,8 @@ module Google
|
|
38
38
|
attr_accessor :quota_project
|
39
39
|
attr_accessor :enable_leader_aware_routing
|
40
40
|
|
41
|
+
attr_reader :universe_domain
|
42
|
+
|
41
43
|
RST_STREAM_INTERNAL_ERROR = "Received RST_STREAM".freeze
|
42
44
|
EOS_INTERNAL_ERROR = "Received unexpected EOS on DATA frame from server".freeze
|
43
45
|
|
@@ -45,11 +47,19 @@ module Google
|
|
45
47
|
# Creates a new Service instance.
|
46
48
|
def initialize project, credentials, quota_project: nil,
|
47
49
|
host: nil, timeout: nil, lib_name: nil, lib_version: nil,
|
48
|
-
enable_leader_aware_routing: nil
|
50
|
+
enable_leader_aware_routing: nil, universe_domain: nil
|
49
51
|
@project = project
|
50
52
|
@credentials = credentials
|
51
53
|
@quota_project = quota_project || (credentials.quota_project_id if credentials.respond_to? :quota_project_id)
|
52
|
-
|
54
|
+
# TODO: This logic is part of UniverseDomainConcerns in gapic-common
|
55
|
+
# but is being copied here because we need to determine the host up
|
56
|
+
# front in order to build a gRPC channel. We should refactor this
|
57
|
+
# somehow to allow this logic to live where it is supposed to.
|
58
|
+
@universe_domain = universe_domain || ENV["GOOGLE_CLOUD_UNIVERSE_DOMAIN"] || "googleapis.com"
|
59
|
+
@host = host ||
|
60
|
+
Google::Cloud::Spanner::V1::Spanner::Client::DEFAULT_ENDPOINT_TEMPLATE.sub(
|
61
|
+
Gapic::UniverseDomainConcerns::ENDPOINT_SUBSTITUTION, @universe_domain
|
62
|
+
)
|
53
63
|
@timeout = timeout
|
54
64
|
@lib_name = lib_name
|
55
65
|
@lib_version = lib_version
|
@@ -74,12 +84,13 @@ module Google
|
|
74
84
|
|
75
85
|
def service
|
76
86
|
return mocked_service if mocked_service
|
77
|
-
@service ||=
|
87
|
+
@service ||=
|
78
88
|
V1::Spanner::Client.new do |config|
|
79
89
|
config.credentials = channel
|
80
90
|
config.quota_project = @quota_project
|
81
91
|
config.timeout = timeout if timeout
|
82
92
|
config.endpoint = host if host
|
93
|
+
config.universe_domain = @universe_domain
|
83
94
|
config.lib_name = lib_name_with_prefix
|
84
95
|
config.lib_version = Google::Cloud::Spanner::VERSION
|
85
96
|
config.metadata = { "google-cloud-resource-prefix" => "projects/#{@project}" }
|
@@ -89,12 +100,13 @@ module Google
|
|
89
100
|
|
90
101
|
def instances
|
91
102
|
return mocked_instances if mocked_instances
|
92
|
-
@instances ||=
|
103
|
+
@instances ||=
|
93
104
|
Admin::Instance::V1::InstanceAdmin::Client.new do |config|
|
94
105
|
config.credentials = channel
|
95
106
|
config.quota_project = @quota_project
|
96
107
|
config.timeout = timeout if timeout
|
97
108
|
config.endpoint = host if host
|
109
|
+
config.universe_domain = @universe_domain
|
98
110
|
config.lib_name = lib_name_with_prefix
|
99
111
|
config.lib_version = Google::Cloud::Spanner::VERSION
|
100
112
|
config.metadata = { "google-cloud-resource-prefix" => "projects/#{@project}" }
|
@@ -104,12 +116,13 @@ module Google
|
|
104
116
|
|
105
117
|
def databases
|
106
118
|
return mocked_databases if mocked_databases
|
107
|
-
@databases ||=
|
119
|
+
@databases ||=
|
108
120
|
Admin::Database::V1::DatabaseAdmin::Client.new do |config|
|
109
121
|
config.credentials = channel
|
110
122
|
config.quota_project = @quota_project
|
111
123
|
config.timeout = timeout if timeout
|
112
124
|
config.endpoint = host if host
|
125
|
+
config.universe_domain = @universe_domain
|
113
126
|
config.lib_name = lib_name_with_prefix
|
114
127
|
config.lib_version = Google::Cloud::Spanner::VERSION
|
115
128
|
config.metadata = { "google-cloud-resource-prefix" => "projects/#{@project}" }
|
@@ -257,12 +270,27 @@ module Google
|
|
257
270
|
end
|
258
271
|
|
259
272
|
def update_database_ddl instance_id, database_id, statements: [],
|
260
|
-
operation_id: nil, call_options: nil
|
273
|
+
operation_id: nil, call_options: nil, descriptor_set: nil
|
274
|
+
bin_data =
|
275
|
+
case descriptor_set
|
276
|
+
when Google::Protobuf::FileDescriptorSet
|
277
|
+
Google::Protobuf::FileDescriptorSet.encode descriptor_set
|
278
|
+
when String
|
279
|
+
File.binread descriptor_set
|
280
|
+
when NilClass
|
281
|
+
nil
|
282
|
+
else
|
283
|
+
raise ArgumentError,
|
284
|
+
"A value of type #{descriptor_set.class} is not supported."
|
285
|
+
end
|
286
|
+
|
287
|
+
proto_descriptors = bin_data unless bin_data.nil?
|
261
288
|
opts = default_options call_options: call_options
|
262
289
|
request = {
|
263
290
|
database: database_path(instance_id, database_id),
|
264
291
|
statements: Array(statements),
|
265
|
-
operation_id: operation_id
|
292
|
+
operation_id: operation_id,
|
293
|
+
proto_descriptors: proto_descriptors
|
266
294
|
}
|
267
295
|
databases.update_database_ddl request, opts
|
268
296
|
end
|
@@ -125,6 +125,7 @@ module Google
|
|
125
125
|
# | `BYTES` | `File`, `IO`, `StringIO`, or similar | |
|
126
126
|
# | `ARRAY` | `Array` | Nested arrays are not supported. |
|
127
127
|
# | `STRUCT` | `Hash`, {Data} | |
|
128
|
+
# | `PROTO` | Determined by proto_fqn | |
|
128
129
|
#
|
129
130
|
# See [Data
|
130
131
|
# types](https://cloud.google.com/spanner/docs/data-definition-language#data_types).
|
data/lib/google/cloud/spanner.rb
CHANGED
@@ -84,6 +84,7 @@ module Google
|
|
84
84
|
# with version.
|
85
85
|
# @param enable_leader_aware_routing [Boolean] Specifies whether Leader
|
86
86
|
# Aware Routing should be enabled. Defaults to true.
|
87
|
+
# @param universe_domain [String] A custom universe domain. Optional.
|
87
88
|
#
|
88
89
|
# @return [Google::Cloud::Spanner::Project]
|
89
90
|
#
|
@@ -95,7 +96,7 @@ module Google
|
|
95
96
|
def self.new project_id: nil, credentials: nil, scope: nil, timeout: nil,
|
96
97
|
endpoint: nil, project: nil, keyfile: nil,
|
97
98
|
emulator_host: nil, lib_name: nil, lib_version: nil,
|
98
|
-
enable_leader_aware_routing: true
|
99
|
+
enable_leader_aware_routing: true, universe_domain: nil
|
99
100
|
project_id ||= project || default_project_id
|
100
101
|
scope ||= configure.scope
|
101
102
|
timeout ||= configure.timeout
|
@@ -104,6 +105,7 @@ module Google
|
|
104
105
|
credentials ||= keyfile
|
105
106
|
lib_name ||= configure.lib_name
|
106
107
|
lib_version ||= configure.lib_version
|
108
|
+
universe_domain ||= configure.universe_domain
|
107
109
|
|
108
110
|
if emulator_host
|
109
111
|
credentials = :this_channel_is_insecure
|
@@ -125,7 +127,7 @@ module Google
|
|
125
127
|
Spanner::Service.new(
|
126
128
|
project_id, credentials, quota_project: configure.quota_project,
|
127
129
|
host: endpoint, timeout: timeout, lib_name: lib_name,
|
128
|
-
lib_version: lib_version,
|
130
|
+
lib_version: lib_version, universe_domain: universe_domain,
|
129
131
|
enable_leader_aware_routing: enable_leader_aware_routing
|
130
132
|
),
|
131
133
|
query_options: configure.query_options
|
data/lib/google-cloud-spanner.rb
CHANGED
@@ -139,10 +139,8 @@ module Google
|
|
139
139
|
end
|
140
140
|
end
|
141
141
|
|
142
|
-
# rubocop:disable Metrics/BlockLength
|
143
|
-
|
144
142
|
# Set the default spanner configuration
|
145
|
-
Google::Cloud.configure.add_config! :spanner do |config|
|
143
|
+
Google::Cloud.configure.add_config! :spanner do |config| # rubocop:disable Metrics/BlockLength
|
146
144
|
default_project = Google::Cloud::Config.deferred do
|
147
145
|
ENV["SPANNER_PROJECT"]
|
148
146
|
end
|
@@ -181,11 +179,10 @@ Google::Cloud.configure.add_config! :spanner do |config|
|
|
181
179
|
config.add_field! :scope, default_scopes, match: [String, Array]
|
182
180
|
config.add_field! :quota_project, nil, match: String
|
183
181
|
config.add_field! :timeout, nil, match: Integer
|
184
|
-
config.add_field! :endpoint,
|
182
|
+
config.add_field! :endpoint, nil, match: String
|
185
183
|
config.add_field! :emulator_host, default_emulator, match: String, allow_nil: true
|
186
184
|
config.add_field! :lib_name, nil, match: String, allow_nil: true
|
187
185
|
config.add_field! :lib_version, nil, match: String, allow_nil: true
|
188
186
|
config.add_field! :query_options, default_query_options, match: Hash, allow_nil: true
|
187
|
+
config.add_field! :universe_domain, nil, match: String, allow_nil: true
|
189
188
|
end
|
190
|
-
|
191
|
-
# rubocop:enable Metrics/BlockLength
|
metadata
CHANGED
@@ -1,15 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: google-cloud-spanner
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 2.
|
4
|
+
version: 2.26.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Mike Moore
|
8
8
|
- Chris Smith
|
9
|
-
autorequire:
|
10
9
|
bindir: bin
|
11
10
|
cert_chain: []
|
12
|
-
date:
|
11
|
+
date: 2025-03-28 00:00:00.000000000 Z
|
13
12
|
dependencies:
|
14
13
|
- !ruby/object:Gem::Dependency
|
15
14
|
name: bigdecimal
|
@@ -45,74 +44,56 @@ dependencies:
|
|
45
44
|
requirements:
|
46
45
|
- - "~>"
|
47
46
|
- !ruby/object:Gem::Version
|
48
|
-
version: '1.
|
47
|
+
version: '1.7'
|
49
48
|
type: :runtime
|
50
49
|
prerelease: false
|
51
50
|
version_requirements: !ruby/object:Gem::Requirement
|
52
51
|
requirements:
|
53
52
|
- - "~>"
|
54
53
|
- !ruby/object:Gem::Version
|
55
|
-
version: '1.
|
54
|
+
version: '1.7'
|
56
55
|
- !ruby/object:Gem::Dependency
|
57
56
|
name: google-cloud-spanner-admin-database-v1
|
58
57
|
requirement: !ruby/object:Gem::Requirement
|
59
58
|
requirements:
|
60
|
-
- - "
|
61
|
-
- !ruby/object:Gem::Version
|
62
|
-
version: 0.19.0
|
63
|
-
- - "<"
|
59
|
+
- - "~>"
|
64
60
|
- !ruby/object:Gem::Version
|
65
|
-
version:
|
61
|
+
version: '1.4'
|
66
62
|
type: :runtime
|
67
63
|
prerelease: false
|
68
64
|
version_requirements: !ruby/object:Gem::Requirement
|
69
65
|
requirements:
|
70
|
-
- - "
|
71
|
-
- !ruby/object:Gem::Version
|
72
|
-
version: 0.19.0
|
73
|
-
- - "<"
|
66
|
+
- - "~>"
|
74
67
|
- !ruby/object:Gem::Version
|
75
|
-
version:
|
68
|
+
version: '1.4'
|
76
69
|
- !ruby/object:Gem::Dependency
|
77
70
|
name: google-cloud-spanner-admin-instance-v1
|
78
71
|
requirement: !ruby/object:Gem::Requirement
|
79
72
|
requirements:
|
80
|
-
- - "
|
81
|
-
- !ruby/object:Gem::Version
|
82
|
-
version: 0.16.0
|
83
|
-
- - "<"
|
73
|
+
- - "~>"
|
84
74
|
- !ruby/object:Gem::Version
|
85
|
-
version:
|
75
|
+
version: '1.6'
|
86
76
|
type: :runtime
|
87
77
|
prerelease: false
|
88
78
|
version_requirements: !ruby/object:Gem::Requirement
|
89
79
|
requirements:
|
90
|
-
- - "
|
91
|
-
- !ruby/object:Gem::Version
|
92
|
-
version: 0.16.0
|
93
|
-
- - "<"
|
80
|
+
- - "~>"
|
94
81
|
- !ruby/object:Gem::Version
|
95
|
-
version:
|
82
|
+
version: '1.6'
|
96
83
|
- !ruby/object:Gem::Dependency
|
97
84
|
name: google-cloud-spanner-v1
|
98
85
|
requirement: !ruby/object:Gem::Requirement
|
99
86
|
requirements:
|
100
|
-
- - "
|
101
|
-
- !ruby/object:Gem::Version
|
102
|
-
version: 0.27.0
|
103
|
-
- - "<"
|
87
|
+
- - "~>"
|
104
88
|
- !ruby/object:Gem::Version
|
105
|
-
version:
|
89
|
+
version: '1.6'
|
106
90
|
type: :runtime
|
107
91
|
prerelease: false
|
108
92
|
version_requirements: !ruby/object:Gem::Requirement
|
109
93
|
requirements:
|
110
|
-
- - "
|
111
|
-
- !ruby/object:Gem::Version
|
112
|
-
version: 0.27.0
|
113
|
-
- - "<"
|
94
|
+
- - "~>"
|
114
95
|
- !ruby/object:Gem::Version
|
115
|
-
version:
|
96
|
+
version: '1.6'
|
116
97
|
description: google-cloud-spanner is the official library for Google Cloud Spanner
|
117
98
|
API.
|
118
99
|
email:
|
@@ -187,7 +168,6 @@ homepage: https://github.com/googleapis/ruby-spanner/blob/main/google-cloud-span
|
|
187
168
|
licenses:
|
188
169
|
- Apache-2.0
|
189
170
|
metadata: {}
|
190
|
-
post_install_message:
|
191
171
|
rdoc_options: []
|
192
172
|
require_paths:
|
193
173
|
- lib
|
@@ -195,15 +175,14 @@ required_ruby_version: !ruby/object:Gem::Requirement
|
|
195
175
|
requirements:
|
196
176
|
- - ">="
|
197
177
|
- !ruby/object:Gem::Version
|
198
|
-
version: '
|
178
|
+
version: '3.0'
|
199
179
|
required_rubygems_version: !ruby/object:Gem::Requirement
|
200
180
|
requirements:
|
201
181
|
- - ">="
|
202
182
|
- !ruby/object:Gem::Version
|
203
183
|
version: '0'
|
204
184
|
requirements: []
|
205
|
-
rubygems_version: 3.5
|
206
|
-
signing_key:
|
185
|
+
rubygems_version: 3.6.5
|
207
186
|
specification_version: 4
|
208
187
|
summary: API Client library for Google Cloud Spanner API
|
209
188
|
test_files: []
|