google-cloud-logging 2.0.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 +7 -0
- data/.yardopts +18 -0
- data/AUTHENTICATION.md +178 -0
- data/CHANGELOG.md +407 -0
- data/CODE_OF_CONDUCT.md +40 -0
- data/CONTRIBUTING.md +188 -0
- data/INSTRUMENTATION.md +71 -0
- data/LICENSE +201 -0
- data/LOGGING.md +32 -0
- data/OVERVIEW.md +321 -0
- data/TROUBLESHOOTING.md +31 -0
- data/lib/google-cloud-logging.rb +161 -0
- data/lib/google/cloud/logging.rb +188 -0
- data/lib/google/cloud/logging/async_writer.rb +513 -0
- data/lib/google/cloud/logging/convert.rb +70 -0
- data/lib/google/cloud/logging/credentials.rb +44 -0
- data/lib/google/cloud/logging/entry.rb +528 -0
- data/lib/google/cloud/logging/entry/http_request.rb +167 -0
- data/lib/google/cloud/logging/entry/list.rb +178 -0
- data/lib/google/cloud/logging/entry/operation.rb +91 -0
- data/lib/google/cloud/logging/entry/source_location.rb +85 -0
- data/lib/google/cloud/logging/errors.rb +101 -0
- data/lib/google/cloud/logging/log/list.rb +156 -0
- data/lib/google/cloud/logging/logger.rb +633 -0
- data/lib/google/cloud/logging/metric.rb +168 -0
- data/lib/google/cloud/logging/metric/list.rb +170 -0
- data/lib/google/cloud/logging/middleware.rb +307 -0
- data/lib/google/cloud/logging/project.rb +838 -0
- data/lib/google/cloud/logging/rails.rb +232 -0
- data/lib/google/cloud/logging/resource.rb +85 -0
- data/lib/google/cloud/logging/resource_descriptor.rb +137 -0
- data/lib/google/cloud/logging/resource_descriptor/list.rb +175 -0
- data/lib/google/cloud/logging/service.rb +239 -0
- data/lib/google/cloud/logging/sink.rb +315 -0
- data/lib/google/cloud/logging/sink/list.rb +168 -0
- data/lib/google/cloud/logging/version.rb +22 -0
- metadata +304 -0
@@ -0,0 +1,167 @@
|
|
1
|
+
# Copyright 2016 Google LLC
|
2
|
+
#
|
3
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
4
|
+
# you may not use this file except in compliance with the License.
|
5
|
+
# You may obtain a copy of the License at
|
6
|
+
#
|
7
|
+
# https://www.apache.org/licenses/LICENSE-2.0
|
8
|
+
#
|
9
|
+
# Unless required by applicable law or agreed to in writing, software
|
10
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
11
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
12
|
+
# See the License for the specific language governing permissions and
|
13
|
+
# limitations under the License.
|
14
|
+
|
15
|
+
|
16
|
+
module Google
|
17
|
+
module Cloud
|
18
|
+
module Logging
|
19
|
+
class Entry
|
20
|
+
##
|
21
|
+
# # Http Request
|
22
|
+
#
|
23
|
+
# HTTP request data associated with a log entry.
|
24
|
+
#
|
25
|
+
# See also {Google::Cloud::Logging::Entry#http_request}.
|
26
|
+
#
|
27
|
+
class HttpRequest
|
28
|
+
##
|
29
|
+
# @private Create an empty HttpRequest object.
|
30
|
+
def initialize; end
|
31
|
+
|
32
|
+
##
|
33
|
+
# The request method. Examples: `"GET"`, `"HEAD"`, `"PUT"`, `"POST"`.
|
34
|
+
# (String)
|
35
|
+
attr_accessor :request_method
|
36
|
+
|
37
|
+
##
|
38
|
+
# @overload method()
|
39
|
+
# Deprecated. Use {#request_method} instead.
|
40
|
+
#
|
41
|
+
# The request method. Examples: `"GET"`, `"HEAD"`, `"PUT"`,
|
42
|
+
# `"POST"`. (String)
|
43
|
+
def method *args
|
44
|
+
# Call Object#method when args are present.
|
45
|
+
return super unless args.empty?
|
46
|
+
|
47
|
+
request_method
|
48
|
+
end
|
49
|
+
|
50
|
+
##
|
51
|
+
# @overload method()
|
52
|
+
# Deprecated. Use {#request_method=} instead.
|
53
|
+
#
|
54
|
+
# The request method. Examples: `"GET"`, `"HEAD"`, `"PUT"`,
|
55
|
+
# `"POST"`. (String)
|
56
|
+
def method= new_request_method
|
57
|
+
self.request_method = new_request_method
|
58
|
+
end
|
59
|
+
|
60
|
+
##
|
61
|
+
# The URL. The scheme (http, https), the host name, the path and the
|
62
|
+
# query portion of the URL that was requested. Example:
|
63
|
+
# `"http://example.com/some/info?color=red"`. (String)
|
64
|
+
attr_accessor :url
|
65
|
+
|
66
|
+
##
|
67
|
+
# The size of the HTTP request message in bytes, including the request
|
68
|
+
# headers and the request body. (Integer)
|
69
|
+
attr_accessor :size
|
70
|
+
|
71
|
+
##
|
72
|
+
# The response code indicating the status of response. Examples:
|
73
|
+
# `200`, `404`. (Integer)
|
74
|
+
attr_accessor :status
|
75
|
+
|
76
|
+
##
|
77
|
+
# The size of the HTTP response message sent back to the client, in
|
78
|
+
# bytes, including the response headers and the response body.
|
79
|
+
# (Integer)
|
80
|
+
attr_accessor :response_size
|
81
|
+
|
82
|
+
##
|
83
|
+
# The user agent sent by the client. Example: `"Mozilla/4.0
|
84
|
+
# (compatible; MSIE 6.0; Windows 98; Q312461; .NET CLR 1.0.3705)"`.
|
85
|
+
# (String)
|
86
|
+
attr_accessor :user_agent
|
87
|
+
|
88
|
+
##
|
89
|
+
# The IP address (IPv4 or IPv6) of the client that issued the HTTP
|
90
|
+
# request. Examples: `"192.168.1.1"`, `"FE80::0202:B3FF:FE1E:8329"`.
|
91
|
+
# (String)
|
92
|
+
attr_accessor :remote_ip
|
93
|
+
|
94
|
+
##
|
95
|
+
# The referer URL of the request, as defined in [HTTP/1.1 Header Field
|
96
|
+
# Definitions](http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html).
|
97
|
+
# (String)
|
98
|
+
attr_accessor :referer
|
99
|
+
|
100
|
+
##
|
101
|
+
# Whether an entity was served from cache (with or without
|
102
|
+
# validation). (Boolean)
|
103
|
+
attr_accessor :cache_hit
|
104
|
+
|
105
|
+
##
|
106
|
+
# Whether the response was validated with the origin server before
|
107
|
+
# being served from cache. This field is only meaningful if
|
108
|
+
# `cache_hit` is `true`. (Boolean)
|
109
|
+
attr_accessor :validated
|
110
|
+
|
111
|
+
##
|
112
|
+
# @private Determines if the HttpRequest has any data.
|
113
|
+
def empty?
|
114
|
+
method.nil? &&
|
115
|
+
url.nil? &&
|
116
|
+
size.nil? &&
|
117
|
+
status.nil? &&
|
118
|
+
response_size.nil? &&
|
119
|
+
user_agent.nil? &&
|
120
|
+
remote_ip.nil? &&
|
121
|
+
referer.nil? &&
|
122
|
+
cache_hit.nil? &&
|
123
|
+
validated.nil?
|
124
|
+
end
|
125
|
+
|
126
|
+
##
|
127
|
+
# @private Exports the HttpRequest to a
|
128
|
+
# Google::Cloud::Logging::Type::HttpRequest object.
|
129
|
+
def to_grpc
|
130
|
+
return nil if empty?
|
131
|
+
Google::Cloud::Logging::Type::HttpRequest.new(
|
132
|
+
request_method: request_method.to_s,
|
133
|
+
request_url: url.to_s,
|
134
|
+
request_size: size.to_i,
|
135
|
+
status: status.to_i,
|
136
|
+
response_size: response_size.to_i,
|
137
|
+
user_agent: user_agent.to_s,
|
138
|
+
remote_ip: remote_ip.to_s,
|
139
|
+
referer: referer.to_s,
|
140
|
+
cache_hit: !(!cache_hit),
|
141
|
+
cache_validated_with_origin_server: !(!validated)
|
142
|
+
)
|
143
|
+
end
|
144
|
+
|
145
|
+
##
|
146
|
+
# @private New HttpRequest from a Google::Cloud::Logging::Type::HttpRequest
|
147
|
+
# object.
|
148
|
+
def self.from_grpc grpc
|
149
|
+
return new if grpc.nil?
|
150
|
+
new.tap do |h|
|
151
|
+
h.request_method = grpc.request_method
|
152
|
+
h.url = grpc.request_url
|
153
|
+
h.size = grpc.request_size
|
154
|
+
h.status = grpc.status
|
155
|
+
h.response_size = grpc.response_size
|
156
|
+
h.user_agent = grpc.user_agent
|
157
|
+
h.remote_ip = grpc.remote_ip
|
158
|
+
h.referer = grpc.referer
|
159
|
+
h.cache_hit = grpc.cache_hit
|
160
|
+
h.validated = grpc.cache_validated_with_origin_server
|
161
|
+
end
|
162
|
+
end
|
163
|
+
end
|
164
|
+
end
|
165
|
+
end
|
166
|
+
end
|
167
|
+
end
|
@@ -0,0 +1,178 @@
|
|
1
|
+
# Copyright 2016 Google LLC
|
2
|
+
#
|
3
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
4
|
+
# you may not use this file except in compliance with the License.
|
5
|
+
# You may obtain a copy of the License at
|
6
|
+
#
|
7
|
+
# https://www.apache.org/licenses/LICENSE-2.0
|
8
|
+
#
|
9
|
+
# Unless required by applicable law or agreed to in writing, software
|
10
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
11
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
12
|
+
# See the License for the specific language governing permissions and
|
13
|
+
# limitations under the License.
|
14
|
+
|
15
|
+
|
16
|
+
require "delegate"
|
17
|
+
|
18
|
+
module Google
|
19
|
+
module Cloud
|
20
|
+
module Logging
|
21
|
+
class Entry
|
22
|
+
##
|
23
|
+
# Entry::List is a special case Array with additional values.
|
24
|
+
class List < DelegateClass(::Array)
|
25
|
+
##
|
26
|
+
# If not empty, indicates that there are more records that match
|
27
|
+
# the request and this value should be passed to continue.
|
28
|
+
attr_accessor :token
|
29
|
+
|
30
|
+
##
|
31
|
+
# @private Create a new Entry::List with an array of Entry instances.
|
32
|
+
def initialize arr = []
|
33
|
+
super arr
|
34
|
+
end
|
35
|
+
|
36
|
+
##
|
37
|
+
# Whether there is a next page of entries.
|
38
|
+
#
|
39
|
+
# @return [Boolean]
|
40
|
+
#
|
41
|
+
# @example
|
42
|
+
# require "google/cloud/logging"
|
43
|
+
#
|
44
|
+
# logging = Google::Cloud::Logging.new
|
45
|
+
#
|
46
|
+
# entries = logging.entries
|
47
|
+
# if entries.next?
|
48
|
+
# next_entries = entries.next
|
49
|
+
# end
|
50
|
+
#
|
51
|
+
def next?
|
52
|
+
!token.nil?
|
53
|
+
end
|
54
|
+
|
55
|
+
##
|
56
|
+
# Retrieve the next page of entries.
|
57
|
+
#
|
58
|
+
# @return [Sink::List]
|
59
|
+
#
|
60
|
+
# @example
|
61
|
+
# require "google/cloud/logging"
|
62
|
+
#
|
63
|
+
# logging = Google::Cloud::Logging.new
|
64
|
+
#
|
65
|
+
# entries = logging.entries
|
66
|
+
# if entries.next?
|
67
|
+
# next_entries = entries.next
|
68
|
+
# end
|
69
|
+
#
|
70
|
+
def next
|
71
|
+
return nil unless next?
|
72
|
+
ensure_service!
|
73
|
+
grpc = @service.list_entries token: token, resources: @resources,
|
74
|
+
filter: @filter, order: @order,
|
75
|
+
max: @max, projects: @projects
|
76
|
+
self.class.from_grpc grpc, @service, resources: @resources,
|
77
|
+
filter: @filter, order: @order,
|
78
|
+
max: @max, projects: @projects
|
79
|
+
end
|
80
|
+
|
81
|
+
##
|
82
|
+
# Retrieves remaining results by repeatedly invoking {#next} until
|
83
|
+
# {#next?} returns `false`. Calls the given block once for each
|
84
|
+
# result, which is passed as the argument to the block.
|
85
|
+
#
|
86
|
+
# An Enumerator is returned if no block is given.
|
87
|
+
#
|
88
|
+
# This method will make repeated API calls until all remaining results
|
89
|
+
# are retrieved. (Unlike `#each`, for example, which merely iterates
|
90
|
+
# over the results returned by a single API call.) Use with caution.
|
91
|
+
#
|
92
|
+
# @param [Integer] request_limit The upper limit of API requests to
|
93
|
+
# make to load all log entries. Default is no limit.
|
94
|
+
# @yield [entry] The block for accessing each log entry.
|
95
|
+
# @yieldparam [Entry] entry The log entry object.
|
96
|
+
#
|
97
|
+
# @return [Enumerator]
|
98
|
+
#
|
99
|
+
# @example Iterating each log entry by passing a block:
|
100
|
+
# require "google/cloud/logging"
|
101
|
+
#
|
102
|
+
# logging = Google::Cloud::Logging.new
|
103
|
+
# entries = logging.entries order: "timestamp desc"
|
104
|
+
#
|
105
|
+
# entries.all do |e|
|
106
|
+
# puts "[#{e.timestamp}] #{e.log_name} #{e.payload.inspect}"
|
107
|
+
# end
|
108
|
+
#
|
109
|
+
# @example Using the enumerator by not passing a block:
|
110
|
+
# require "google/cloud/logging"
|
111
|
+
#
|
112
|
+
# logging = Google::Cloud::Logging.new
|
113
|
+
# entries = logging.entries order: "timestamp desc"
|
114
|
+
#
|
115
|
+
# all_payloads = entries.all.map do |entry|
|
116
|
+
# entry.payload
|
117
|
+
# end
|
118
|
+
#
|
119
|
+
# @example Limit the number of API calls made:
|
120
|
+
# require "google/cloud/logging"
|
121
|
+
#
|
122
|
+
# logging = Google::Cloud::Logging.new
|
123
|
+
# entries = logging.entries order: "timestamp desc"
|
124
|
+
#
|
125
|
+
# entries.all(request_limit: 10) do |e|
|
126
|
+
# puts "[#{e.timestamp}] #{e.log_name} #{e.payload.inspect}"
|
127
|
+
# end
|
128
|
+
#
|
129
|
+
def all request_limit: nil
|
130
|
+
request_limit = request_limit.to_i if request_limit
|
131
|
+
unless block_given?
|
132
|
+
return enum_for :all, request_limit: request_limit
|
133
|
+
end
|
134
|
+
results = self
|
135
|
+
loop do
|
136
|
+
results.each { |r| yield r }
|
137
|
+
if request_limit
|
138
|
+
request_limit -= 1
|
139
|
+
break if request_limit < 0
|
140
|
+
end
|
141
|
+
break unless results.next?
|
142
|
+
results = results.next
|
143
|
+
end
|
144
|
+
end
|
145
|
+
|
146
|
+
##
|
147
|
+
# @private New Entry::List from a
|
148
|
+
# Google::Cloud::Logging::V2::ListLogEntryResponse object.
|
149
|
+
def self.from_grpc grpc_list, service, resources: nil, filter: nil,
|
150
|
+
order: nil, max: nil, projects: nil
|
151
|
+
entries = new(Array(grpc_list.entries).map do |grpc_entry|
|
152
|
+
Entry.from_grpc grpc_entry
|
153
|
+
end)
|
154
|
+
token = grpc_list.next_page_token
|
155
|
+
token = nil if token == "".freeze
|
156
|
+
entries.instance_variable_set :@token, token
|
157
|
+
entries.instance_variable_set :@service, service
|
158
|
+
entries.instance_variable_set :@projects, projects
|
159
|
+
entries.instance_variable_set :@resources, resources
|
160
|
+
entries.instance_variable_set :@filter, filter
|
161
|
+
entries.instance_variable_set :@order, order
|
162
|
+
entries.instance_variable_set :@max, max
|
163
|
+
entries
|
164
|
+
end
|
165
|
+
|
166
|
+
protected
|
167
|
+
|
168
|
+
##
|
169
|
+
# @private Raise an error unless an active connection to the service
|
170
|
+
# is available.
|
171
|
+
def ensure_service!
|
172
|
+
raise "Must have active connection to service" unless @service
|
173
|
+
end
|
174
|
+
end
|
175
|
+
end
|
176
|
+
end
|
177
|
+
end
|
178
|
+
end
|
@@ -0,0 +1,91 @@
|
|
1
|
+
# Copyright 2016 Google LLC
|
2
|
+
#
|
3
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
4
|
+
# you may not use this file except in compliance with the License.
|
5
|
+
# You may obtain a copy of the License at
|
6
|
+
#
|
7
|
+
# https://www.apache.org/licenses/LICENSE-2.0
|
8
|
+
#
|
9
|
+
# Unless required by applicable law or agreed to in writing, software
|
10
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
11
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
12
|
+
# See the License for the specific language governing permissions and
|
13
|
+
# limitations under the License.
|
14
|
+
|
15
|
+
|
16
|
+
module Google
|
17
|
+
module Cloud
|
18
|
+
module Logging
|
19
|
+
class Entry
|
20
|
+
##
|
21
|
+
# # Operation
|
22
|
+
#
|
23
|
+
# Additional information about a potentially long-running operation with
|
24
|
+
# which a log entry is associated.
|
25
|
+
#
|
26
|
+
# See also {Google::Cloud::Logging::Entry#operation}.
|
27
|
+
#
|
28
|
+
class Operation
|
29
|
+
##
|
30
|
+
# @private Create an empty Operation object.
|
31
|
+
def initialize; end
|
32
|
+
|
33
|
+
##
|
34
|
+
# An arbitrary operation identifier. Log entries with the same
|
35
|
+
# identifier are assumed to be part of the same operation.
|
36
|
+
attr_accessor :id
|
37
|
+
|
38
|
+
##
|
39
|
+
# An arbitrary producer identifier. The combination of `id` and
|
40
|
+
# `producer` must be globally unique. Examples for `producer`:
|
41
|
+
# `"MyDivision.MyBigCompany.com"`,
|
42
|
+
# `"github.com/MyProject/MyApplication"`.
|
43
|
+
attr_accessor :producer
|
44
|
+
|
45
|
+
##
|
46
|
+
# Set this to `true` if this is the first log entry in the operation.
|
47
|
+
attr_accessor :first
|
48
|
+
|
49
|
+
##
|
50
|
+
# Set this to `true` if this is the last log entry in the operation.
|
51
|
+
attr_accessor :last
|
52
|
+
|
53
|
+
##
|
54
|
+
# @private Determines if the Operation has any data.
|
55
|
+
def empty?
|
56
|
+
id.nil? &&
|
57
|
+
producer.nil? &&
|
58
|
+
first.nil? &&
|
59
|
+
last.nil?
|
60
|
+
end
|
61
|
+
|
62
|
+
##
|
63
|
+
# @private Exports the Operation to a
|
64
|
+
# Google::Cloud::Logging::V2::LogEntryOperation object.
|
65
|
+
def to_grpc
|
66
|
+
return nil if empty?
|
67
|
+
Google::Cloud::Logging::V2::LogEntryOperation.new(
|
68
|
+
id: id.to_s,
|
69
|
+
producer: producer.to_s,
|
70
|
+
first: !(!first),
|
71
|
+
last: !(!last)
|
72
|
+
)
|
73
|
+
end
|
74
|
+
|
75
|
+
##
|
76
|
+
# @private New Google::Cloud::Logging::Entry::Operation from a
|
77
|
+
# Google::Cloud::Logging::V2::LogEntryOperation object.
|
78
|
+
def self.from_grpc grpc
|
79
|
+
return new if grpc.nil?
|
80
|
+
new.tap do |o|
|
81
|
+
o.id = grpc.id
|
82
|
+
o.producer = grpc.producer
|
83
|
+
o.first = grpc.first
|
84
|
+
o.last = grpc.last
|
85
|
+
end
|
86
|
+
end
|
87
|
+
end
|
88
|
+
end
|
89
|
+
end
|
90
|
+
end
|
91
|
+
end
|
@@ -0,0 +1,85 @@
|
|
1
|
+
# Copyright 2017 Google LLC
|
2
|
+
#
|
3
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
4
|
+
# you may not use this file except in compliance with the License.
|
5
|
+
# You may obtain a copy of the License at
|
6
|
+
#
|
7
|
+
# https://www.apache.org/licenses/LICENSE-2.0
|
8
|
+
#
|
9
|
+
# Unless required by applicable law or agreed to in writing, software
|
10
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
11
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
12
|
+
# See the License for the specific language governing permissions and
|
13
|
+
# limitations under the License.
|
14
|
+
|
15
|
+
|
16
|
+
module Google
|
17
|
+
module Cloud
|
18
|
+
module Logging
|
19
|
+
class Entry
|
20
|
+
##
|
21
|
+
# # SourceLocation
|
22
|
+
#
|
23
|
+
# Additional information about the source code location that produced
|
24
|
+
# the log entry.
|
25
|
+
#
|
26
|
+
# See also {Google::Cloud::Logging::Entry#source_location}.
|
27
|
+
#
|
28
|
+
class SourceLocation
|
29
|
+
##
|
30
|
+
# @private Create an empty SourceLocation object.
|
31
|
+
def initialize; end
|
32
|
+
|
33
|
+
##
|
34
|
+
# Source file name. Depending on the runtime environment, this might
|
35
|
+
# be a simple name or a fully-qualified name. Optional.
|
36
|
+
attr_accessor :file
|
37
|
+
|
38
|
+
##
|
39
|
+
# Line within the source file. 1-based; `0` indicates no line number
|
40
|
+
# available. Optional.
|
41
|
+
attr_accessor :line
|
42
|
+
|
43
|
+
##
|
44
|
+
# Human-readable name of the function or method being invoked, with
|
45
|
+
# optional context such as the class or package name. This information
|
46
|
+
# may be used in contexts such as the logs viewer, where a file and
|
47
|
+
# line number are less meaningful. Optional.
|
48
|
+
attr_accessor :function
|
49
|
+
|
50
|
+
##
|
51
|
+
# @private Determines if the SourceLocation has any data.
|
52
|
+
def empty?
|
53
|
+
file.nil? &&
|
54
|
+
line.nil? &&
|
55
|
+
function.nil?
|
56
|
+
end
|
57
|
+
|
58
|
+
##
|
59
|
+
# @private Exports the SourceLocation to a
|
60
|
+
# Google::Cloud::Logging::V2::LogEntrySourceLocation object.
|
61
|
+
def to_grpc
|
62
|
+
return nil if empty?
|
63
|
+
Google::Cloud::Logging::V2::LogEntrySourceLocation.new(
|
64
|
+
file: file.to_s,
|
65
|
+
line: line,
|
66
|
+
function: function.to_s
|
67
|
+
)
|
68
|
+
end
|
69
|
+
|
70
|
+
##
|
71
|
+
# @private New Google::Cloud::Logging::Entry::SourceLocation from a
|
72
|
+
# Google::Cloud::Logging::V2::LogEntrySourceLocation object.
|
73
|
+
def self.from_grpc grpc
|
74
|
+
return new if grpc.nil?
|
75
|
+
new.tap do |o|
|
76
|
+
o.file = grpc.file
|
77
|
+
o.line = grpc.line
|
78
|
+
o.function = grpc.function
|
79
|
+
end
|
80
|
+
end
|
81
|
+
end
|
82
|
+
end
|
83
|
+
end
|
84
|
+
end
|
85
|
+
end
|