Skip to content

Commit

Permalink
Add instrumentation for AsyncTransaction
Browse files Browse the repository at this point in the history
  • Loading branch information
TimPansino committed Jul 27, 2023
1 parent 6b7fc79 commit e04ec6f
Show file tree
Hide file tree
Showing 3 changed files with 138 additions and 0 deletions.
5 changes: 5 additions & 0 deletions newrelic/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -2344,6 +2344,11 @@ def _process_module_builtin_defaults():
"newrelic.hooks.datastore_firestore",
"instrument_google_cloud_firestore_v1_transaction",
)
_process_module_definition(
"google.cloud.firestore_v1.async_transaction",
"newrelic.hooks.datastore_firestore",
"instrument_google_cloud_firestore_v1_async_transaction",
)

_process_module_definition(
"ariadne.asgi",
Expand Down
11 changes: 11 additions & 0 deletions newrelic/hooks/datastore_firestore.py
Original file line number Diff line number Diff line change
Expand Up @@ -237,3 +237,14 @@ def instrument_google_cloud_firestore_v1_transaction(module):
wrap_datastore_trace(
module, "Transaction.%s" % method, product="Firestore", target=None, operation=operation
)


def instrument_google_cloud_firestore_v1_async_transaction(module):
if hasattr(module, "AsyncTransaction"):
class_ = module.AsyncTransaction
for method in ("_commit", "_rollback"):
if hasattr(class_, method):
operation = method[1:] # Trim leading underscore
wrap_datastore_trace(
module, "AsyncTransaction.%s" % method, product="Firestore", target=None, operation=operation
)
122 changes: 122 additions & 0 deletions tests/datastore_firestore/test_async_transaction.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,122 @@
# Copyright 2010 New Relic, Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import pytest

from testing_support.validators.validate_transaction_metrics import validate_transaction_metrics
from newrelic.api.background_task import background_task
from testing_support.validators.validate_database_duration import (
validate_database_duration,
)


@pytest.fixture(autouse=True)
def sample_data(collection, reset_firestore):
# reset_firestore must be run before, not after this fixture
for x in range(1, 4):
collection.add({"x": x}, "doc%d" % x)


async def _exercise_async_transaction_commit(async_client, async_collection):
from google.cloud.firestore_v1.async_transaction import async_transactional

@async_transactional
async def _exercise(async_transaction):
# get a DocumentReference
with pytest.raises(TypeError): # get is currently broken. It attempts to await an async_generator instead of consuming it.
[_ async for _ in async_transaction.get(async_collection.document("doc1"))]

# get a Query
with pytest.raises(TypeError): # get is currently broken. It attempts to await an async_generator instead of consuming it.
async_query = async_collection.select("x").where(field_path="x", op_string=">", value=2)
assert len([_ async for _ in async_transaction.get(async_query)]) == 1

# get_all on a list of DocumentReferences
with pytest.raises(TypeError): # get_all is currently broken. It attempts to await an async_generator instead of consuming it.
all_docs = async_transaction.get_all([async_collection.document("doc%d" % x) for x in range(1, 4)])
assert len([_ async for _ in all_docs]) == 3

# set and delete methods
async_transaction.set(async_collection.document("doc2"), {"x": 0})
async_transaction.delete(async_collection.document("doc3"))

await _exercise(async_client.transaction())
assert len([_ async for _ in async_collection.list_documents()]) == 2


async def _exercise_async_transaction_rollback(async_client, async_collection):
from google.cloud.firestore_v1.async_transaction import async_transactional

@async_transactional
async def _exercise(async_transaction):
# set and delete methods
async_transaction.set(async_collection.document("doc2"), {"x": 99})
async_transaction.delete(async_collection.document("doc1"))
raise RuntimeError()

with pytest.raises(RuntimeError):
await _exercise(async_client.transaction())
assert len([_ async for _ in async_collection.list_documents()]) == 3


def test_firestore_async_transaction_commit(loop, async_client, async_collection):
_test_scoped_metrics = [
("Datastore/operation/Firestore/commit", 1),
# ("Datastore/operation/Firestore/get_all", 2),
# ("Datastore/statement/Firestore/%s/stream" % async_collection.id, 1),
("Datastore/statement/Firestore/%s/list_documents" % async_collection.id, 1),
]

_test_rollup_metrics = [
# ("Datastore/operation/Firestore/stream", 1),
("Datastore/operation/Firestore/list_documents", 1),
("Datastore/all", 2), # Should be 5 if not for broken APIs
("Datastore/allOther", 2),
]
@validate_database_duration()
@validate_transaction_metrics(
"test_firestore_async_transaction",
scoped_metrics=_test_scoped_metrics,
rollup_metrics=_test_rollup_metrics,
background_task=True,
)
@background_task(name="test_firestore_async_transaction")
def _test():
loop.run_until_complete(_exercise_async_transaction_commit(async_client, async_collection))

_test()


def test_firestore_async_transaction_rollback(loop, async_client, async_collection):
_test_scoped_metrics = [
("Datastore/operation/Firestore/rollback", 1),
("Datastore/statement/Firestore/%s/list_documents" % async_collection.id, 1),
]

_test_rollup_metrics = [
("Datastore/operation/Firestore/list_documents", 1),
("Datastore/all", 2),
("Datastore/allOther", 2),
]
# @validate_database_duration()
@validate_transaction_metrics(
"test_firestore_async_transaction",
scoped_metrics=_test_scoped_metrics,
rollup_metrics=_test_rollup_metrics,
background_task=True,
)
@background_task(name="test_firestore_async_transaction")
def _test():
loop.run_until_complete(_exercise_async_transaction_rollback(async_client, async_collection))

_test()

0 comments on commit e04ec6f

Please sign in to comment.