-
Notifications
You must be signed in to change notification settings - Fork 67
/
Copy pathlib_aws_sdk_php.c
938 lines (829 loc) · 32.1 KB
/
lib_aws_sdk_php.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
/*
* Copyright 2024 New Relic Corporation. All rights reserved.
* SPDX-License-Identifier: Apache-2.0
*/
/*
* Functions relating to instrumenting the AWS-SDK-PHP.
* https://github.com/aws/aws-sdk-php
*/
#include "php_agent.h"
#include "php_call.h"
#include "php_hash.h"
#include "php_wrapper.h"
#include "fw_hooks.h"
#include "fw_support.h"
#include "util_logging.h"
#include "lib_aws_sdk_php.h"
#define PHP_PACKAGE_NAME "aws/aws-sdk-php"
#define AWS_LAMBDA_ARN_REGEX \
"(arn:(aws[a-zA-Z-]*)?:lambda:)?" \
"((?<region>[a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\\d{1}):)?" \
"((?<accountId>\\d{12}):)?" \
"(function:)?" \
"(?<functionName>[a-zA-Z0-9-\\.]+)" \
"(:(?<qualifier>\\$LATEST|[a-zA-Z0-9-]+))?"
#if ZEND_MODULE_API_NO >= ZEND_8_1_X_API_NO /* PHP8.1+ */
/* Service instrumentation only supported above PHP 8.1+*/
/*
* Note: For SQS, the command_arg_array will contain the following arrays seen
below:
//clang-format off
$result = $client->receiveMessage(array(
// QueueUrl is required
'QueueUrl' => 'string',
'AttributeNames' => array('string', ... ),
'MessageAttributeNames' => array('string', ... ),
'MaxNumberOfMessages' => integer,
'VisibilityTimeout' => integer,
'WaitTimeSeconds' => integer,
));
$result = $client->sendMessage(array(
// QueueUrl is required
'QueueUrl' => 'string',
// MessageBody is required
'MessageBody' => 'string',
'DelaySeconds' => integer,
'MessageAttributes' => array(
// Associative array of custom 'String' key names
'String' => array(
'StringValue' => 'string',
'BinaryValue' => 'string',
'StringListValues' => array('string', ... ),
'BinaryListValues' => array('string', ... ),
// DataType is required
'DataType' => 'string',
),
// ... repeated
),
));
$result = $client->sendMessageBatch(array(
// QueueUrl is required
'QueueUrl' => 'string',
// Entries is required
'Entries' => array(
array(
// Id is required
'Id' => 'string',
// MessageBody is required
'MessageBody' => 'string',
'DelaySeconds' => integer,
'MessageAttributes' => array(
// Associative array of custom 'String' key names
'String' => array(
'StringValue' => 'string',
'BinaryValue' => 'string',
'StringListValues' => array('string', ... ),
'BinaryListValues' => array('string', ... ),
// DataType is required
'DataType' => 'string',
),
// ... repeated
),
),
// ... repeated
),
));
//clang-format on
*/
void nr_lib_aws_sdk_php_sqs_handle(nr_segment_t* auto_segment,
char* command_name_string,
size_t command_name_len,
NR_EXECUTE_PROTO) {
char* command_arg_value = NULL;
nr_segment_t* message_segment = NULL;
nr_segment_message_params_t message_params = {
.library = SQS_LIBRARY_NAME,
.destination_type = NR_MESSAGE_DESTINATION_TYPE_QUEUE,
.messaging_system = AWS_SQS_MESSAGING_SERVICE,
};
nr_segment_cloud_attrs_t cloud_attrs = {0};
if (NULL == auto_segment) {
return;
}
if (NULL == command_name_string || 0 == command_name_len) {
return;
}
#define AWS_COMMAND_IS(CMD) \
(command_name_len == (sizeof(CMD) - 1) && nr_streq(CMD, command_name_string))
/* Determine if we instrument this command. */
if (AWS_COMMAND_IS("sendMessageBatch")) {
message_params.message_action = NR_SPANKIND_PRODUCER;
} else if (AWS_COMMAND_IS("sendMessage")) {
message_params.message_action = NR_SPANKIND_PRODUCER;
} else if (AWS_COMMAND_IS("receiveMessage")) {
message_params.message_action = NR_SPANKIND_CONSUMER;
} else {
/* Nothing to do here so exit. */
return;
}
#undef AWS_COMMAND_IS
/*
* By this point, it's been determined that this call will be instrumented so
* only create the segment now, grab the parent segment start time, add our
* special segment attributes/metrics then close the newly created segment.
*/
message_segment = nr_segment_start(NRPRG(txn), NULL, NULL);
if (NULL == message_segment) {
return;
}
/* re-use start time from auto_segment started in func_begin */
message_segment->start_time = auto_segment->start_time;
cloud_attrs.aws_operation = command_name_string;
command_arg_value = nr_lib_aws_sdk_php_get_command_arg_value(
AWS_SDK_PHP_SQSCLIENT_QUEUEURL_ARG, NR_EXECUTE_ORIG_ARGS);
/*
* nr_lib_aws_sdk_php_sqs_parse_queueurl requires a modifiable string to
* populate message_params and cloud_attrs.
*/
nr_lib_aws_sdk_php_sqs_parse_queueurl(command_arg_value, &message_params,
&cloud_attrs);
/* Add cloud attributes, if available. */
nr_segment_traces_add_cloud_attributes(message_segment, &cloud_attrs);
/* Now end the instrumented segment as a message segment. */
nr_segment_message_end(&message_segment, &message_params);
nr_free(command_arg_value);
}
void nr_lib_aws_sdk_php_sqs_parse_queueurl(
char* sqs_queueurl,
nr_segment_message_params_t* message_params,
nr_segment_cloud_attrs_t* cloud_attrs) {
char* region = NULL;
char* queue_name = NULL;
char* account_id = NULL;
char* queueurl_pointer = NULL;
if (NULL == sqs_queueurl || NULL == message_params || NULL == cloud_attrs) {
return;
}
/*
* AWS QueueUrl has a very specific format.
* The QueueUrl we are looking for will be of the following format:
* queueUrl =
* 'https://sqs.REGION_NAME.amazonaws.com/ACCOUNT_ID_NAME/SQS_QUEUE_NAME'
* where REGION_NAME, ACCOUNT_ID_NAME, and SQS_QUEUE_NAME are the acutal
* values such as: queueUrl =
* 'https://sqs.us-east-2.amazonaws.com/123456789012/my_amazing_queue'
* If we are unable to match any part of this, the whole decode is suspect and
* all values are discarded.
*
* Due to the overhead involved in escaping the original buffer, creating a
* regex, matching a regex, destroying a regex, this method was chosen as a
* more performant option because it's a very limited pattern.
*/
queueurl_pointer = sqs_queueurl;
/*
* Find the pattern of the AWS queueurl that should immediately precede the
* region.
*/
if (0
!= strncmp(queueurl_pointer, AWS_QUEUEURL_PREFIX,
AWS_QUEUEURL_PREFIX_LEN)) {
/* Malformed queueurl, we can't decode this. */
return;
}
/*
* Find the start of the region. It follows the 12 chars of 'https://sqs.'
* and continues until the next '.' It is safe to move the pointer along at
* this point since we just verified the prefix exists.
*/
queueurl_pointer += AWS_QUEUEURL_PREFIX_LEN;
if (nr_strempty(queueurl_pointer)) {
/* Malformed queueurl, we can't decode this. */
return;
}
region = queueurl_pointer;
/* Find the end of the region. */
queueurl_pointer = nr_strchr(queueurl_pointer, '.');
if (NULL == queueurl_pointer) {
/* Malformed queueurl, we can't decode this. */
return;
}
*queueurl_pointer = '\0';
/*
* Move the pointer along. Again, we found a valid '.' so moving the pointer
* beyond that point should be safe and give us either more string or the end
* of the string.
*/
queueurl_pointer += 1;
if (nr_strempty(queueurl_pointer)) {
/* Malformed queueurl, we can't decode this. */
return;
}
/* Move past the next pattern to find the start of the account id. */
if (0
!= strncmp(queueurl_pointer, AWS_QUEUEURL_AWS_POSTFIX,
AWS_QUEUEURL_AWS_POSTFIX_LEN)) {
/* Malformed queueurl, we can't decode this. */
return;
}
/*
* Move the pointer along. Since we found a valid pattern match moving the
* pointer beyond that point should be safe and give us either more string or
* the end of the string.
*/
queueurl_pointer += AWS_QUEUEURL_AWS_POSTFIX_LEN;
if (nr_strempty(queueurl_pointer)) {
/* Malformed queueurl, we can't decode this. */
return;
}
/* If it's not an empty string, we've found the start of the account_id*/
account_id = queueurl_pointer;
/* Find the end of account id which goes until the next forward slash. */
queueurl_pointer = nr_strchr(queueurl_pointer, '/');
if (NULL == queueurl_pointer) {
/* Malformed queueurl, we can't decode this. */
return;
}
*queueurl_pointer = '\0';
/* Move the pointer along. */
queueurl_pointer += 1;
if (nr_strempty(queueurl_pointer)) {
/* Malformed queueurl, we can't decode this. */
return;
}
/* This should be the start of the start of the queuename.*/
queue_name = queueurl_pointer;
/*
* Almost done. At this point, the string should only have queue name left.
* Let's check if there's another slash, if it isn't followed by empty string,
* the queueurl is malformed.
*/
queueurl_pointer = nr_strchr(queueurl_pointer, '/');
if (NULL != queueurl_pointer) {
*queueurl_pointer = '\0';
/* Let's check if it's followed by empty string */
*queueurl_pointer += 1;
if (!nr_strempty(queueurl_pointer)) {
/* Malformed queueurl, we can't decode this. */
return;
}
}
/*
* SQS entity relationship requires: messaging.system, cloud.region,
* cloud.account.id, messaging.destination.name
*/
message_params->destination_name = queue_name;
cloud_attrs->cloud_account_id = account_id;
cloud_attrs->cloud_region = region;
}
void nr_lib_aws_sdk_php_lambda_handle(nr_segment_t* auto_segment,
char* command_name_string,
size_t command_name_len,
NR_EXECUTE_PROTO) {
nr_segment_t* external_segment = NULL;
zval** retval_ptr = NR_GET_RETURN_VALUE_PTR;
nr_segment_cloud_attrs_t cloud_attrs = {.cloud_platform = "aws_lambda"};
if (NULL == auto_segment) {
return;
}
if (NULL == command_name_string || 0 == command_name_len) {
return;
}
if (NULL == *retval_ptr) {
/* Do not instrument when an exception has happened */
return;
}
#define AWS_COMMAND_IS(CMD) \
(command_name_len == (sizeof(CMD) - 1) && nr_streq(CMD, command_name_string))
/* Determine if we instrument this command. */
if (AWS_COMMAND_IS("invoke")) {
/* reconstruct the ARN */
nr_aws_sdk_lambda_client_invoke_parse_args(NR_EXECUTE_ORIG_ARGS,
&cloud_attrs);
} else {
return;
}
#undef AWS_COMMAND_IS
/*
* By this point, it's been determined that this call will be instrumented so
* only create the segment now, grab the parent segment start time, add our
* special segment attributes/metrics then close the newly created segment.
*/
external_segment = nr_segment_start(NRPRG(txn), NULL, NULL);
if (NULL == external_segment) {
nr_free(cloud_attrs.cloud_resource_id);
return;
}
/* re-use start time from auto_segment started in func_begin */
external_segment->start_time = auto_segment->start_time;
cloud_attrs.aws_operation = command_name_string;
/* end the segment */
nr_segment_traces_add_cloud_attributes(external_segment, &cloud_attrs);
nr_segment_external_params_t external_params = {.library = "aws_sdk"};
zval* data = nr_php_get_zval_object_property(*retval_ptr, "data");
if (nr_php_is_zval_valid_array(data)) {
zval* status_code = nr_php_zend_hash_find(Z_ARRVAL_P(data), "StatusCode");
if (nr_php_is_zval_valid_integer(status_code)) {
external_params.status = Z_LVAL_P(status_code);
}
zval* metadata = nr_php_zend_hash_find(Z_ARRVAL_P(data), "@metadata");
if (NULL != metadata && IS_REFERENCE == Z_TYPE_P(metadata)) {
metadata = Z_REFVAL_P(metadata);
}
if (nr_php_is_zval_valid_array(metadata)) {
zval* uri = nr_php_zend_hash_find(Z_ARRVAL_P(metadata), "effectiveUri");
if (nr_php_is_zval_non_empty_string(uri)) {
external_params.uri = Z_STRVAL_P(uri);
}
}
}
nr_segment_external_end(&external_segment, &external_params);
nr_free(cloud_attrs.cloud_resource_id);
}
/* This stores the compiled regex to parse AWS ARNs. The compilation happens
* when it is first needed and is destroyed in mshutdown
*/
static nr_regex_t* aws_arn_regex;
static void nr_aws_sdk_compile_regex(void) {
aws_arn_regex = nr_regex_create(AWS_LAMBDA_ARN_REGEX, 0, 0);
}
void nr_aws_sdk_mshutdown(void) {
nr_regex_destroy(&aws_arn_regex);
}
void nr_aws_sdk_lambda_client_invoke_parse_args(
NR_EXECUTE_PROTO,
nr_segment_cloud_attrs_t* cloud_attrs) {
zval* call_args = nr_php_get_user_func_arg(2, NR_EXECUTE_ORIG_ARGS);
zval* this_obj = NR_PHP_USER_FN_THIS();
char* arn = NULL;
char* function_name = NULL;
char* region = NULL;
zval* region_zval = NULL;
char* qualifier = NULL;
char* accountID = NULL;
bool using_account_id_ini = false;
/* verify arguments */
if (!nr_php_is_zval_valid_array(call_args)) {
return;
}
zval* lambda_args = nr_php_zend_hash_index_find(Z_ARRVAL_P(call_args), 0);
if (!nr_php_is_zval_valid_array(lambda_args)) {
return;
}
zval* lambda_name
= nr_php_zend_hash_find(Z_ARRVAL_P(lambda_args), "FunctionName");
if (!nr_php_is_zval_non_empty_string(lambda_name)) {
return;
}
/* Ensure regex exists */
if (NULL == aws_arn_regex) {
nr_aws_sdk_compile_regex();
}
/* Extract all information possible from the passed lambda name via regex */
nr_regex_substrings_t* matches = nr_regex_match_capture(
aws_arn_regex, Z_STRVAL_P(lambda_name), Z_STRLEN_P(lambda_name));
function_name = nr_regex_substrings_get_named(matches, "functionName");
accountID = nr_regex_substrings_get_named(matches, "accountId");
region = nr_regex_substrings_get_named(matches, "region");
qualifier = nr_regex_substrings_get_named(matches, "qualifier");
/* supplement missing information with API calls */
if (nr_strempty(function_name)) {
/*
* Cannot get the needed data. Function name is required in the
* argument, so this won't happen in normal operation
*/
nr_free(function_name);
nr_free(accountID);
nr_free(region);
nr_free(qualifier);
nr_regex_substrings_destroy(&matches);
return;
}
if (nr_strempty(accountID)) {
nr_free(accountID);
accountID = NRINI(aws_account_id);
using_account_id_ini = true;
}
if (nr_strempty(region)) {
zend_class_entry* base_class = NULL;
if (NULL != execute_data->func
&& NULL != execute_data->func->common.scope) {
base_class = execute_data->func->common.scope;
}
region_zval = nr_php_get_zval_object_property_with_class(
this_obj, base_class, "region");
if (nr_php_is_zval_valid_string(region_zval)) {
/*
* In this case, region is likely to be NULL, but could be an empty
* string instead, so we must free
*/
nr_free(region);
region = Z_STRVAL_P(region_zval);
}
}
if (!nr_strempty(accountID) && !nr_strempty(region)) {
/* construct the ARN */
if (!nr_strempty(qualifier)) {
arn = nr_formatf("arn:aws:lambda:%s:%s:function:%s:%s", region, accountID,
function_name, qualifier);
} else {
arn = nr_formatf("arn:aws:lambda:%s:%s:function:%s", region, accountID,
function_name);
}
/* Attach the ARN */
cloud_attrs->cloud_resource_id = arn;
}
nr_regex_substrings_destroy(&matches);
nr_free(function_name);
if (!using_account_id_ini) {
nr_free(accountID);
}
/* if region_zval is a valid string, we have already freed region */
if (!nr_php_is_zval_valid_string(region_zval)) {
nr_free(region);
}
nr_free(qualifier);
}
char* nr_lib_aws_sdk_php_get_command_arg_value(char* command_arg_name,
NR_EXECUTE_PROTO) {
zval* param_array = NULL;
zval* command_arg_array = NULL;
char* command_arg_value = NULL;
if (NULL == command_arg_name) {
return NULL;
}
/* To extract the Aws/AwsClient::__call $argument, we get the second arg. */
param_array = nr_php_arg_get(2, NR_EXECUTE_ORIG_ARGS);
if (nr_php_is_zval_valid_array(param_array)) {
/* The first element in param_array is an array of parameters. */
command_arg_array = nr_php_zend_hash_index_find(Z_ARRVAL_P(param_array), 0);
if (nr_php_is_zval_valid_array(command_arg_array)) {
zval* queueurl_arg = nr_php_zend_hash_find(Z_ARRVAL_P(command_arg_array),
command_arg_name);
if (nr_php_is_zval_non_empty_string(queueurl_arg)) {
command_arg_value = nr_strdup(Z_STRVAL_P(queueurl_arg));
}
}
}
nr_php_arg_release(¶m_array);
return command_arg_value;
}
void nr_lib_aws_sdk_php_dynamodb_set_params(
nr_segment_datastore_params_t* datastore_params,
nr_segment_cloud_attrs_t* cloud_attrs,
NR_EXECUTE_PROTO) {
zval* endpoint_zval = NULL;
zval* region_zval = NULL;
zval* host_zval = NULL;
zval* port_zval = NULL;
zval* this_obj = NULL;
zend_function* func = NULL;
zend_class_entry* base_class = NULL;
char* table_name = NULL;
char* account_id = NULL;
if (NULL == datastore_params || NULL == cloud_attrs) {
return;
}
this_obj = NR_PHP_USER_FN_THIS();
func = nr_php_execute_function(NR_EXECUTE_ORIG_ARGS);
if (NULL == this_obj || NULL == func) {
return;
}
if (NULL != func->common.scope) {
base_class = func->common.scope;
region_zval = nr_php_get_zval_object_property_with_class(
this_obj, base_class, "region");
if (nr_php_is_zval_non_empty_string(region_zval)) {
cloud_attrs->cloud_region = Z_STRVAL_P(region_zval);
}
endpoint_zval = nr_php_get_zval_object_property_with_class(
this_obj, base_class, "endpoint");
if (nr_php_is_zval_valid_object(endpoint_zval)) {
host_zval = nr_php_get_zval_object_property(endpoint_zval, "host");
if (nr_php_is_zval_non_empty_string(host_zval)) {
datastore_params->instance->host = Z_STRVAL_P(host_zval);
/* Only try to get a port if we have a valid host. */
port_zval = nr_php_get_zval_object_property(endpoint_zval, "port");
if (nr_php_is_zval_valid_integer(port_zval)) {
/* Must be freed by caller */
datastore_params->instance->port_path_or_id
= nr_formatf(NR_INT64_FMT, Z_LVAL_P(port_zval));
} else {
/* In case where host was found but port was not, spec says return
* unknown for port. */
datastore_params->instance->port_path_or_id = nr_strdup("unknown");
}
}
}
}
if (NULL == datastore_params->instance->host) {
/* Unable to retrieve the endpoint, go with AWS defaults. */
datastore_params->instance->host = AWS_SDK_PHP_DYNAMODBCLIENT_DEFAULT_HOST;
/* Need to strdup because the calling function will free it. */
datastore_params->instance->port_path_or_id
= nr_strdup(AWS_SDK_PHP_DYNAMODBCLIENT_DEFAULT_PORT);
}
table_name = nr_lib_aws_sdk_php_get_command_arg_value(
AWS_SDK_PHP_DYNAMODBCLIENT_TABLENAME_ARG, NR_EXECUTE_ORIG_ARGS);
if (!nr_strempty(table_name)) {
/* Must be freed by caller */
datastore_params->collection = table_name;
}
if (!nr_strempty(NRINI(aws_account_id))) {
account_id = NRINI(aws_account_id);
}
if (NULL != datastore_params->collection && NULL != account_id
&& NULL != cloud_attrs->cloud_region) {
/* Must be freed by caller */
cloud_attrs->cloud_resource_id = nr_formatf(
"arn:aws:dynamodb:%s:%s:table/%s", cloud_attrs->cloud_region,
account_id, datastore_params->collection);
}
}
void nr_lib_aws_sdk_php_dynamodb_handle(nr_segment_t* auto_segment,
char* command_name_string,
size_t command_name_len,
NR_EXECUTE_PROTO) {
nr_segment_t* datastore_segment = NULL;
nr_segment_cloud_attrs_t cloud_attrs = {0};
nr_datastore_instance_t instance = {0};
nr_segment_datastore_params_t datastore_params = {
.db_system = AWS_SDK_PHP_DYNAMODBCLIENT_DATASTORE_SYSTEM,
.datastore = {
.type = NR_DATASTORE_DYNAMODB,
},
.instance = &instance,
.callbacks = {
.backtrace = nr_php_backtrace_callback,
},
};
if (NULL == auto_segment) {
return;
}
if (NULL == command_name_string || 0 == command_name_len) {
return;
}
#define AWS_COMMAND_IS(CMD) \
(command_name_len == (sizeof(CMD) - 1) && nr_streq(CMD, command_name_string))
/* Determine if we instrument this command. */
if (AWS_COMMAND_IS("createTable")) {
datastore_params.operation = AWS_SDK_PHP_DYNAMODBCLIENT_CREATE_TABLE;
} else if (AWS_COMMAND_IS("deleteItem")) {
datastore_params.operation = AWS_SDK_PHP_DYNAMODBCLIENT_DELETE_ITEM;
} else if (AWS_COMMAND_IS("deleteTable")) {
datastore_params.operation = AWS_SDK_PHP_DYNAMODBCLIENT_DELETE_TABLE;
} else if (AWS_COMMAND_IS("getItem")) {
datastore_params.operation = AWS_SDK_PHP_DYNAMODBCLIENT_GET_ITEM;
} else if (AWS_COMMAND_IS("putItem")) {
datastore_params.operation = AWS_SDK_PHP_DYNAMODBCLIENT_PUT_ITEM;
} else if (AWS_COMMAND_IS("query")) {
datastore_params.operation = AWS_SDK_PHP_DYNAMODBCLIENT_QUERY;
} else if (AWS_COMMAND_IS("scan")) {
datastore_params.operation = AWS_SDK_PHP_DYNAMODBCLIENT_SCAN;
} else if (AWS_COMMAND_IS("updateItem")) {
datastore_params.operation = AWS_SDK_PHP_DYNAMODBCLIENT_UPDATE_ITEM;
} else {
/* Nothing to do here so exit. */
return;
}
#undef AWS_COMMAND_IS
/*
* nr_lib_aws_sdk_php_dynamodb_set_params sets:
* the cloud_attrs->region and cloud_resource_id(needs to be freed)
* datastore->instance host and port_path_or_id(needs to be freed)
* datastore->collection (needs to be freed)
*/
nr_lib_aws_sdk_php_dynamodb_set_params(&datastore_params, &cloud_attrs,
NR_EXECUTE_ORIG_ARGS);
/*
* By this point, the datastore params are decoded, grab the parent segment
* start time, add the special segment attributes/metrics then close the newly
* created segment.
*/
datastore_segment = nr_segment_start(NRPRG(txn), NULL, NULL);
if (NULL == datastore_segment) {
return;
}
/* re-use start time from auto_segment started in func_begin */
datastore_segment->start_time = auto_segment->start_time;
cloud_attrs.aws_operation = command_name_string;
/* Add cloud attributes, if available. */
nr_segment_traces_add_cloud_attributes(datastore_segment, &cloud_attrs);
/* Now end the instrumented segment as a message segment. */
nr_segment_datastore_end(&datastore_segment, &datastore_params);
nr_free(datastore_params.collection);
nr_free(cloud_attrs.cloud_resource_id);
nr_free(instance.port_path_or_id);
}
/*
* For Aws/AwsClient::__call see
* https://github.com/aws/aws-sdk-php/blob/master/src/AwsClientInterface.php
* ALL
* client commands are handled by this function, so it is the start and end of
* any command. Creates and executes a command for an operation by name.
* When a class command isn't explicitly created as a function, the __call class
* handles the invocation. This means all AWS Client Service commands are
* handled by this call. Any invocation starts when this function starts, and
* ends when it ends. This function decodes the command name, determines the
* appropriate args, decodes the args, generates a guzzle request to send to the
* AWS Service, gets the guzzle response from the AWS Service, and bundles that
* response into an AswResult to return.
*
* @param string $name Name of the command to execute.
* @param array $arguments Arguments to pass to the getCommand method.
*
* @return ResultInterface
* @throws \Exception
*/
NR_PHP_WRAPPER(nr_aws_client_call) {
(void)wraprec;
zval* command_name = NULL;
const char* klass = NULL;
char* command_name_string = NULL;
char* real_class_and_command = NULL;
nr_segment_t* segment = NULL;
zend_class_entry* class_entry = NULL;
int klass_len = 0;
class_entry = Z_OBJCE_P(nr_php_execute_scope(execute_data));
if (NULL == class_entry) {
goto end;
}
klass = nr_php_class_entry_name(class_entry);
if (NULL == klass) {
goto end;
}
/* Get the arg command_name. */
command_name = nr_php_arg_get(1, NR_EXECUTE_ORIG_ARGS);
if (!nr_php_is_zval_non_empty_string(command_name)) {
goto end;
}
command_name_string = Z_STRVAL_P(command_name);
klass_len = nr_php_class_entry_name_length(class_entry);
#define AWS_CLASS_IS(KLASS, SHORT_KLASS) \
(klass_len == (sizeof(KLASS) - 1) \
&& nr_striendswith(klass, klass_len, SHORT_KLASS, sizeof(SHORT_KLASS) - 1))
if (AWS_CLASS_IS("Aws\\Sqs\\SqsClient", "SqsClient")) {
nr_lib_aws_sdk_php_sqs_handle(auto_segment, command_name_string,
Z_STRLEN_P(command_name),
NR_EXECUTE_ORIG_ARGS);
} else if (AWS_CLASS_IS("Aws\\Lambda\\LambdaClient", "LambdaClient")) {
nr_lib_aws_sdk_php_lambda_handle(auto_segment, command_name_string,
Z_STRLEN_P(command_name),
NR_EXECUTE_ORIG_ARGS);
} else if (AWS_CLASS_IS("Aws\\DynamoDb\\DynamoDbClient", "DynamoDbClient")) {
nr_lib_aws_sdk_php_dynamodb_handle(auto_segment, command_name_string,
Z_STRLEN_P(command_name),
NR_EXECUTE_ORIG_ARGS);
}
#undef AWS_CLASS_IS
/*
* Since we have klass and command_name, we can give the calling segment
* a more meaningful name than Aws/AwsClient::__call We can decode it to
* Aws/CALLING_CLASS_NAME/CALLING_CLASS_CLIENT::CALLING_CLASS_COMMAND
*
* EX: Aws\\Sqs\\SqsClient::sendMessage
*/
if (NULL != auto_segment) {
real_class_and_command
= nr_formatf("Custom/%s::%s", klass, command_name_string);
nr_segment_set_name(auto_segment, real_class_and_command);
nr_free(real_class_and_command);
}
end:
/* Release the command_name. */
nr_php_arg_release(&command_name);
}
NR_PHP_WRAPPER_END
#endif /* PHP >= 8.1*/
/*
* In a normal course of events, the following line will always work
* zend_eval_string("Aws\\Sdk::VERSION;", &retval, "Get AWS Version")
* By the time we have detected the existence of the aws-sdk-php and with
* default composer project settings, it is callable even from
* nr_aws_sdk_php_enable which will automatically load the class if it isn't
* loaded yet and then evaluate the string. In the rare case that files
* are not loaded via autoloader and/or have non-default composer classload
* settings, if the class is not found, PHP 8.2+ will generate an
* error whenever it cannot find a class which must be caught. Calling this
* from nr_aws_sdk_php_enable would allow the sdk version value to be set only
* once. To avoid the VERY unlikely but not impossible fatal error, we need to
* wrap the call in a try/catch block and make it a lambda so that we avoid
* fatal errors.
*/
void nr_lib_aws_sdk_php_handle_version() {
char* version = NULL;
zval retval;
int result = FAILURE;
/*
* The following block initializes nr_aws_sdk_version to the empty string.
* If it is able to extract the version, nr_aws_sdk_version is set to that.
* Nothing is needed in the catch block.
* The final return will either return a proper version or an empty string.
*/
result = zend_eval_string(
"(function() {"
" $nr_aws_sdk_version = '';"
" try {"
" $nr_aws_sdk_version = Aws\\Sdk::VERSION;"
" } catch (Throwable $e) {"
" }"
" return $nr_aws_sdk_version;"
"})();",
&retval, "Get nr_aws_sdk_version");
/* See if we got a non-empty/non-null string for version. */
if (SUCCESS == result) {
if (nr_php_is_zval_non_empty_string(&retval)) {
version = Z_STRVAL(retval);
}
}
if (NRINI(vulnerability_management_package_detection_enabled)) {
/* Add php package to transaction */
nr_txn_add_php_package(NRPRG(txn), PHP_PACKAGE_NAME, version);
}
nr_txn_suggest_package_supportability_metric(NRPRG(txn), PHP_PACKAGE_NAME,
version);
zval_dtor(&retval);
}
void nr_lib_aws_sdk_php_add_supportability_service_metric(
const char* service_name) {
/* total MAX metric name length per agent-specs */
char buf[MAX_METRIC_NAME_LEN];
char* cp = NULL;
if (nr_strempty(service_name)) {
return;
}
if (NULL == NRPRG(txn)) {
return;
}
cp = buf;
nr_strcpy(cp, PHP_AWS_SDK_SERVICE_NAME_METRIC_PREFIX);
cp += PHP_AWS_SDK_SERVICE_NAME_METRIC_PREFIX_LEN - 1;
nr_strlcpy(cp, service_name, MAX_AWS_SERVICE_NAME_LEN);
nrm_force_add(NRPRG(txn) ? NRTXN(unscoped_metrics) : 0, buf, 0);
}
/*
* AwsClient::parseClass
* This is called from the base AwsClient class for every client associated
* with a service during client initialization.
* parseClass already computes the service name for internal use, so we don't
* need to store it, we just need to snag it from the return value as it goes
* through the client initialization process.
*/
NR_PHP_WRAPPER(nr_create_aws_service_metric) {
(void)wraprec;
zval** ret_val_ptr = NULL;
ret_val_ptr = NR_GET_RETURN_VALUE_PTR;
NR_PHP_WRAPPER_CALL;
if (NULL != ret_val_ptr && nr_php_is_zval_valid_array(*ret_val_ptr)) {
/* obtain ret_val_ptr[0] which contains the service name */
zval* service_name
= nr_php_zend_hash_index_find(Z_ARRVAL_P(*ret_val_ptr), 0);
if (nr_php_is_zval_non_empty_string(service_name)) {
nr_lib_aws_sdk_php_add_supportability_service_metric(
Z_STRVAL_P(service_name));
}
}
}
NR_PHP_WRAPPER_END
/*
* The ideal file to begin immediate detection of the aws-sdk is:
* aws-sdk-php/src/functions.php
* Unfortunately, Php8.2+ and composer autoload leads to the
* file being optimized directly and not loaded.
*
* Options considered:
*
* 1. for PHP8.2, and only optimizable libraries, when encountering autoload.php
* files, ask the file what includes it added and check against only the
* optimizable library. Small overhead incurred when encountering an autoload
* file, but detects aws-sdk-php immediately before any sdk code executes
* (changes needed for this are detailed in the original PR)
* 2. use a file that gets called later and only when AwsClient.php file is
* called. It's called later and we'll miss some instrumentation, but if we're
* only ever going to be interested in Client calls anyway, maybe that's ok?
* Doesn't detect Sdk.php (optimized out) so when customers only use that or
* when they use it first, we will not instrument it. This only detects when a
* Client is called to use a service so potentially misses out on other
* instrumentation and misses out when customers use the aws-sdk-php but use
* non-SDK way to interact with the service (possibly with redis/memcached).
* This way is definitely the least complex and lowest overhead and less
* complexity means lower risk as well.
* 3. Directly add the wrappers to the hash map. With potentially 50ish clients
* to wrap, this will add overhead to every hash map lookup. Currently
* implemented option is 2, use the AwsClient.php as this is our main focus.
* This means until a call to an Aws/AwsClient function,
* all calls including aws\sdk calls are ignored.
*
* Version detection will be called directly from Aws\Sdk.php
*/
void nr_aws_sdk_php_enable() {
/*
* Set the UNKNOWN package first, so it doesn't overwrite what we find with
* nr_lib_aws_sdk_php_handle_version.
*/
if (NRINI(vulnerability_management_package_detection_enabled)) {
nr_txn_add_php_package(NRPRG(txn), PHP_PACKAGE_NAME,
PHP_PACKAGE_VERSION_UNKNOWN);
}
/* Extract the version for aws-sdk 3+ */
nr_lib_aws_sdk_php_handle_version();
/* Called when initializing all Clients */
nr_php_wrap_user_function(NR_PSTR("Aws\\AwsClient::parseClass"),
nr_create_aws_service_metric);
#if ZEND_MODULE_API_NO >= ZEND_8_1_X_API_NO /* PHP8.1+ */
/* We only support instrumentation above PHP 8.1 */
/* Called when a service command is issued from a Client */
nr_php_wrap_user_function_before_after_clean(
NR_PSTR("Aws\\AwsClient::__call"), NULL, nr_aws_client_call,
nr_aws_client_call);
#endif
}