-
Notifications
You must be signed in to change notification settings - Fork 6
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
Azure load tests #1665
base: main
Are you sure you want to change the base?
Azure load tests #1665
Changes from 7 commits
9c64c15
6ef84a2
e90565c
790abd7
0f55439
34a783a
54cfc1b
7f8b16b
7896fd0
022e9b5
8be024f
fcd9c16
be3a898
9a013b0
565d2b6
b085490
d8ae1a4
c905fe7
467cc3b
fbdf77e
d3aa1da
283abd1
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -4,6 +4,7 @@ | |
import urllib.parse | ||
import urllib.request | ||
import uuid | ||
import os | ||
|
||
from locust import FastHttpUser, between, events, task | ||
from locust.runners import MasterRunner | ||
|
@@ -19,6 +20,8 @@ | |
result_request_body = None | ||
auth_request_body = None | ||
|
||
in_azure = os.getenv('TEST_RUN_NAME') is not None | ||
|
||
|
||
class SampleUser(FastHttpUser): | ||
# Each task gets called randomly, but the number next to '@task' denotes | ||
|
@@ -118,6 +121,10 @@ def test_start(environment): | |
|
||
@events.quitting.add_listener | ||
def assert_stats(environment): | ||
if in_azure: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ensure that the environment-specific paths and configurations, such as file paths and environment variables, are documented and validated to prevent runtime errors in different environments. [medium] |
||
# don't evaluate this in Azure because we want the locust process to succeed and Azure does its own test criteria checking | ||
return | ||
|
||
if environment.stats.total.fail_ratio > 0.01: | ||
logging.error("Test failed due to failure ratio > 1%") | ||
environment.process_exit_code = 1 | ||
|
@@ -131,22 +138,38 @@ def assert_stats(environment): | |
def get_auth_request_body(): | ||
# set up the sample request body for the auth endpoint | ||
# using a valid test token found in the mock_credentials directory | ||
auth_scope = "report-stream" | ||
with open("mock_credentials/report-stream-valid-token.jwt") as f: | ||
auth_token = f.read() | ||
|
||
# TODO - notes/clarification on 2 different creds, plus expiration date of jwt | ||
# TODO - do we want to TF the tests? If yes which envs? In CDC envs, may need to adjust IP allow list on app. Also set as private endpoints in test config? | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is a good question. My assumption is that we don't want load test bombarding CDC environments unless they're okay with that. If they aren't, this could just be an internal running load test. However, I feel if we only run it in internal, it loses value since that would mean we're only running these until we pass the application off to the CDC. |
||
# TODO - currently in Azure we're specifying a version for the key vault item (so if it gets updated, we'll be referencing an old version) - do we want to change this? | ||
if in_azure: | ||
auth_token = os.getenv("trusted-intermediary-valid-token-jwt") | ||
else: | ||
with open("mock_credentials/trusted-intermediary-valid-token.jwt") as f: | ||
auth_token = f.read() | ||
|
||
params = urllib.parse.urlencode( | ||
{"scope": auth_scope, "client_assertion": auth_token.strip()} | ||
{"scope": "trusted-intermediary", "client_assertion": auth_token.strip()} | ||
) | ||
|
||
return params.encode("utf-8") | ||
|
||
|
||
def get_order_fhir_message(): | ||
# read the sample request body for the orders endpoint | ||
with open("examples/Test/e2e/orders/002_ORM_O01_short.fhir", "r") as f: | ||
file_path = "002_ORM_O01_short.fhir" | ||
if not in_azure: | ||
file_path = "examples/Test/e2e/orders/" + file_path | ||
|
||
with open(file_path, "r") as f: | ||
return f.read() | ||
|
||
|
||
def get_result_fhir_message(): | ||
# read the sample request body for the results endpoint | ||
with open("examples/Test/e2e/results/001_ORU_R01_short.fhir", "r") as f: | ||
file_path = "001_ORU_R01_short.fhir" | ||
if not in_azure: | ||
file_path = "examples/Test/e2e/results/" + file_path | ||
|
||
with open(file_path, "r") as f: | ||
return f.read() |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -13,6 +13,7 @@ | |
import java.nio.file.attribute.PosixFilePermissions; | ||
import java.util.ArrayList; | ||
import java.util.Arrays; | ||
import java.util.HashMap; | ||
import java.util.HashSet; | ||
import java.util.List; | ||
import java.util.Map; | ||
|
@@ -28,6 +29,8 @@ | |
public class ApplicationContext { | ||
|
||
protected static final Map<Class<?>, Object> OBJECT_MAP = new ConcurrentHashMap<>(); | ||
protected static final InheritableThreadLocal<Map<Class<?>, Object>> THREAD_OBJECT_MAP = | ||
new InheritableThreadLocal<>(); | ||
protected static final Map<String, String> TEST_ENV_VARS = new ConcurrentHashMap<>(); | ||
protected static final Set<Object> IMPLEMENTATIONS = new HashSet<>(); | ||
|
||
|
@@ -40,7 +43,34 @@ public static void register(Class<?> clazz, Object implementation) { | |
IMPLEMENTATIONS.add(implementation.getClass()); | ||
} | ||
|
||
public static void registerForThread(Class<?> clazz, Object implementation) { | ||
Map<Class<?>, Object> threadObjectMap = THREAD_OBJECT_MAP.get(); | ||
if (threadObjectMap == null) { | ||
threadObjectMap = new HashMap<>(); | ||
} | ||
|
||
threadObjectMap.put(clazz, implementation); | ||
|
||
THREAD_OBJECT_MAP.set(threadObjectMap); | ||
|
||
// The implementation may never have had anything injected into it | ||
// (e.g. it wasn't part of the bootstrapping implementations registered into the | ||
// ApplicationContext), | ||
// so inject into the implementation now. | ||
injectIntoNonSingleton(implementation); | ||
} | ||
|
||
public static void clearThreadRegistrations() { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Consider using a more robust thread-local cleanup strategy. The current use of THREAD_OBJECT_MAP.remove() might lead to memory leaks if not all paths that add to the thread-local store also ensure to clear it after use. [important] |
||
THREAD_OBJECT_MAP.remove(); | ||
} | ||
|
||
public static <T> T getImplementation(Class<T> clazz) { | ||
// check the thread local map first | ||
Map<Class<?>, Object> threadObjectMap = THREAD_OBJECT_MAP.get(); | ||
if (threadObjectMap != null && threadObjectMap.containsKey(clazz)) { | ||
return (T) threadObjectMap.get(clazz); | ||
} | ||
|
||
T object = (T) OBJECT_MAP.get(clazz); | ||
|
||
if (object == null) { | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would it be better to do something like
Boolean.parseBoolean(request.getHeaders().get("load-test"))
? This handles casing and null safetyThere was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I like that.