Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add a timestamp parameter to the logger. #2860

Merged
merged 4 commits into from
Dec 15, 2016
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
55 changes: 41 additions & 14 deletions logging/google/cloud/logging/logger.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
import json

from google.protobuf.json_format import MessageToJson
from google.cloud._helpers import _datetime_to_rfc3339


class Logger(object):
Expand Down Expand Up @@ -92,7 +93,7 @@ def batch(self, client=None):

def _make_entry_resource(self, text=None, info=None, message=None,
labels=None, insert_id=None, severity=None,
http_request=None):
http_request=None, timestamp=None):
"""Return a log entry resource of the appropriate type.

Helper for :meth:`log_text`, :meth:`log_struct`, and :meth:`log_proto`.
Expand Down Expand Up @@ -121,6 +122,9 @@ def _make_entry_resource(self, text=None, info=None, message=None,
:param http_request: (optional) info about HTTP request associated with
the entry

:type timestamp: :class:`datetime.datetime`
:param timestamp: (optional) timestamp of event being logged.

:rtype: dict
:returns: The JSON resource created.
"""
Expand Down Expand Up @@ -155,10 +159,13 @@ def _make_entry_resource(self, text=None, info=None, message=None,
if http_request is not None:
resource['httpRequest'] = http_request

if timestamp is not None:
resource['timestamp'] = _datetime_to_rfc3339(timestamp)

return resource

def log_text(self, text, client=None, labels=None, insert_id=None,
severity=None, http_request=None):
severity=None, http_request=None, timestamp=None):
"""API call: log a text message via a POST request

See:
Expand All @@ -184,15 +191,18 @@ def log_text(self, text, client=None, labels=None, insert_id=None,
:type http_request: dict
:param http_request: (optional) info about HTTP request associated with
the entry

:type timestamp: :class:`datetime.datetime`
:param timestamp: (optional) timestamp of event being logged.
"""
client = self._require_client(client)
entry_resource = self._make_entry_resource(
text=text, labels=labels, insert_id=insert_id, severity=severity,
http_request=http_request)
http_request=http_request, timestamp=timestamp)
client.logging_api.write_entries([entry_resource])

def log_struct(self, info, client=None, labels=None, insert_id=None,
severity=None, http_request=None):
severity=None, http_request=None, timestamp=None):
"""API call: log a structured message via a POST request

See:
Expand All @@ -218,15 +228,18 @@ def log_struct(self, info, client=None, labels=None, insert_id=None,
:type http_request: dict
:param http_request: (optional) info about HTTP request associated with
the entry.

:type timestamp: :class:`datetime.datetime`
:param timestamp: (optional) timestamp of event being logged.
"""
client = self._require_client(client)
entry_resource = self._make_entry_resource(
info=info, labels=labels, insert_id=insert_id, severity=severity,
http_request=http_request)
http_request=http_request, timestamp=timestamp)
client.logging_api.write_entries([entry_resource])

def log_proto(self, message, client=None, labels=None, insert_id=None,
severity=None, http_request=None):
severity=None, http_request=None, timestamp=None):
"""API call: log a protobuf message via a POST request

See:
Expand All @@ -252,11 +265,14 @@ def log_proto(self, message, client=None, labels=None, insert_id=None,
:type http_request: dict
:param http_request: (optional) info about HTTP request associated with
the entry.

:type timestamp: :class:`datetime.datetime`
:param timestamp: (optional) timestamp of event being logged.
"""
client = self._require_client(client)
entry_resource = self._make_entry_resource(
message=message, labels=labels, insert_id=insert_id,
severity=severity, http_request=http_request)
severity=severity, http_request=http_request, timestamp=timestamp)
client.logging_api.write_entries([entry_resource])

def delete(self, client=None):
Expand Down Expand Up @@ -340,7 +356,7 @@ def __exit__(self, exc_type, exc_val, exc_tb):
self.commit()

def log_text(self, text, labels=None, insert_id=None, severity=None,
http_request=None):
http_request=None, timestamp=None):
"""Add a text entry to be logged during :meth:`commit`.

:type text: str
Expand All @@ -358,12 +374,15 @@ def log_text(self, text, labels=None, insert_id=None, severity=None,
:type http_request: dict
:param http_request: (optional) info about HTTP request associated with
the entry.

:type timestamp: :class:`datetime.datetime`
:param timestamp: (optional) timestamp of event being logged.
"""
self.entries.append(
('text', text, labels, insert_id, severity, http_request))
('text', text, labels, insert_id, severity, http_request, timestamp))

def log_struct(self, info, labels=None, insert_id=None, severity=None,
http_request=None):
http_request=None, timestamp=None):
"""Add a struct entry to be logged during :meth:`commit`.

:type info: dict
Expand All @@ -381,12 +400,15 @@ def log_struct(self, info, labels=None, insert_id=None, severity=None,
:type http_request: dict
:param http_request: (optional) info about HTTP request associated with
the entry.

:type timestamp: :class:`datetime.datetime`
:param timestamp: (optional) timestamp of event being logged.
"""
self.entries.append(
('struct', info, labels, insert_id, severity, http_request))
('struct', info, labels, insert_id, severity, http_request, timestamp))

def log_proto(self, message, labels=None, insert_id=None, severity=None,
http_request=None):
http_request=None, timestamp=None):
"""Add a protobuf entry to be logged during :meth:`commit`.

:type message: protobuf message
Expand All @@ -404,9 +426,12 @@ def log_proto(self, message, labels=None, insert_id=None, severity=None,
:type http_request: dict
:param http_request: (optional) info about HTTP request associated with
the entry.

:type timestamp: :class:`datetime.datetime`
:param timestamp: (optional) timestamp of event being logged.
"""
self.entries.append(
('proto', message, labels, insert_id, severity, http_request))
('proto', message, labels, insert_id, severity, http_request, timestamp))

def commit(self, client=None):
"""Send saved log entries as a single API call.
Expand All @@ -427,7 +452,7 @@ def commit(self, client=None):
kwargs['labels'] = self.logger.labels

entries = []
for entry_type, entry, labels, iid, severity, http_req in self.entries:
for entry_type, entry, labels, iid, severity, http_req, timestamp in self.entries:
if entry_type == 'text':
info = {'textPayload': entry}
elif entry_type == 'struct':
Expand All @@ -446,6 +471,8 @@ def commit(self, client=None):
info['severity'] = severity
if http_req is not None:
info['httpRequest'] = http_req
if timestamp is not None:
info['timestamp'] = timestamp
entries.append(info)

client.logging_api.write_entries(entries, **kwargs)
Expand Down
30 changes: 17 additions & 13 deletions logging/unit_tests/test_logger.py
Original file line number Diff line number Diff line change
Expand Up @@ -454,7 +454,7 @@ def test_log_text_defaults(self):
batch = self._make_one(logger, client=client)
batch.log_text(TEXT)
self.assertEqual(batch.entries,
[('text', TEXT, None, None, None, None)])
[('text', TEXT, None, None, None, None, None)])

def test_log_text_explicit(self):
TEXT = 'This is the entry text'
Expand All @@ -469,13 +469,14 @@ def test_log_text_explicit(self):
'requestUrl': URI,
'status': STATUS,
}
TIMESTAMP = '2016-10-12T15:01:23.045123456Z'
client = _Client(project=self.PROJECT, connection=_make_credentials())
logger = _Logger()
batch = self._make_one(logger, client=client)
batch.log_text(TEXT, labels=LABELS, insert_id=IID, severity=SEVERITY,
http_request=REQUEST)
http_request=REQUEST, timestamp=TIMESTAMP)
self.assertEqual(batch.entries,
[('text', TEXT, LABELS, IID, SEVERITY, REQUEST)])
[('text', TEXT, LABELS, IID, SEVERITY, REQUEST, TIMESTAMP)])

def test_log_struct_defaults(self):
STRUCT = {'message': 'Message text', 'weather': 'partly cloudy'}
Expand All @@ -484,7 +485,7 @@ def test_log_struct_defaults(self):
batch = self._make_one(logger, client=client)
batch.log_struct(STRUCT)
self.assertEqual(batch.entries,
[('struct', STRUCT, None, None, None, None)])
[('struct', STRUCT, None, None, None, None, None)])

This comment was marked as spam.


def test_log_struct_explicit(self):
STRUCT = {'message': 'Message text', 'weather': 'partly cloudy'}
Expand All @@ -499,13 +500,14 @@ def test_log_struct_explicit(self):
'requestUrl': URI,
'status': STATUS,
}
TIMESTAMP = '2016-10-12T15:01:23.045123456Z'
client = _Client(project=self.PROJECT, connection=_make_credentials())
logger = _Logger()
batch = self._make_one(logger, client=client)
batch.log_struct(STRUCT, labels=LABELS, insert_id=IID,
severity=SEVERITY, http_request=REQUEST)
severity=SEVERITY, http_request=REQUEST, timestamp=TIMESTAMP)
self.assertEqual(batch.entries,

This comment was marked as spam.

[('struct', STRUCT, LABELS, IID, SEVERITY, REQUEST)])
[('struct', STRUCT, LABELS, IID, SEVERITY, REQUEST, TIMESTAMP)])

This comment was marked as spam.


def test_log_proto_defaults(self):
from google.protobuf.struct_pb2 import Struct, Value
Expand All @@ -515,7 +517,7 @@ def test_log_proto_defaults(self):
batch = self._make_one(logger, client=client)
batch.log_proto(message)
self.assertEqual(batch.entries,
[('proto', message, None, None, None, None)])
[('proto', message, None, None, None, None, None)])

def test_log_proto_explicit(self):
from google.protobuf.struct_pb2 import Struct, Value
Expand All @@ -531,13 +533,14 @@ def test_log_proto_explicit(self):
'requestUrl': URI,
'status': STATUS,
}
TIMESTAMP = '2016-10-12T15:01:23.045123456Z'
client = _Client(project=self.PROJECT, connection=_make_credentials())
logger = _Logger()
batch = self._make_one(logger, client=client)
batch.log_proto(message, labels=LABELS, insert_id=IID,
severity=SEVERITY, http_request=REQUEST)
severity=SEVERITY, http_request=REQUEST, timestamp=TIMESTAMP)

This comment was marked as spam.

self.assertEqual(batch.entries,
[('proto', message, LABELS, IID, SEVERITY, REQUEST)])
[('proto', message, LABELS, IID, SEVERITY, REQUEST, TIMESTAMP)])

This comment was marked as spam.


def test_commit_w_invalid_entry_type(self):
logger = _Logger()
Expand Down Expand Up @@ -681,20 +684,21 @@ def test_context_mgr_failure(self):
'requestUrl': URI,
'status': STATUS,
}
TIMESTAMP = '2016-10-12T15:01:23.045123456Z'
message = Struct(fields={'foo': Value(bool_value=True)})
client = _Client(project=self.PROJECT)
api = client.logging_api = _DummyLoggingAPI()
logger = _Logger()
UNSENT = [
('text', TEXT, None, IID, None, None),
('struct', STRUCT, None, None, SEVERITY, None),
('proto', message, LABELS, None, None, REQUEST),
('text', TEXT, None, IID, None, None, TIMESTAMP),
('struct', STRUCT, None, None, SEVERITY, None, None),
('proto', message, LABELS, None, None, REQUEST, None),
]
batch = self._make_one(logger, client=client)

try:
with batch as other:
other.log_text(TEXT, insert_id=IID)
other.log_text(TEXT, insert_id=IID, timestamp=TIMESTAMP)
other.log_struct(STRUCT, severity=SEVERITY)
other.log_proto(message, labels=LABELS, http_request=REQUEST)
raise _Bugout()
Expand Down