@@ -298,6 +298,13 @@ static void flb_loki_kv_exit(struct flb_loki *ctx)
298
298
mk_list_foreach_safe (head , tmp , & ctx -> structured_metadata_list ) {
299
299
kv = mk_list_entry (head , struct flb_loki_kv , _head );
300
300
301
+ /* unlink and destroy */
302
+ mk_list_del (& kv -> _head );
303
+ flb_loki_kv_destroy (kv );
304
+ }
305
+ mk_list_foreach_safe (head , tmp , & ctx -> structured_metadata_map_keys_list ) {
306
+ kv = mk_list_entry (head , struct flb_loki_kv , _head );
307
+
301
308
/* unlink and destroy */
302
309
mk_list_del (& kv -> _head );
303
310
flb_loki_kv_destroy (kv );
@@ -416,6 +423,93 @@ static void pack_kv(struct flb_loki *ctx,
416
423
}
417
424
}
418
425
426
+ /*
427
+ * Similar to pack_kv above, except will only use msgpack_objects of type
428
+ * MSGPACK_OBJECT_MAP, and will iterate over the keys adding each entry as a
429
+ * separate item. Non-string map values are serialised to JSON, as Loki requires
430
+ * all values to be strings.
431
+ */
432
+ static void pack_maps (struct flb_loki * ctx ,
433
+ msgpack_packer * mp_pck ,
434
+ char * tag , int tag_len ,
435
+ msgpack_object * map ,
436
+ struct flb_mp_map_header * mh ,
437
+ struct mk_list * list )
438
+ {
439
+ struct mk_list * head ;
440
+ struct flb_loki_kv * kv ;
441
+
442
+ msgpack_object * start_key ;
443
+ msgpack_object * out_key ;
444
+ msgpack_object * out_val ;
445
+
446
+ msgpack_object_map accessed_map ;
447
+ uint32_t accessed_map_index ;
448
+ msgpack_object_kv accessed_map_kv ;
449
+
450
+ char * accessed_map_val_json ;
451
+
452
+ mk_list_foreach (head , list ) {
453
+ /* get the flb_loki_kv for this iteration of the loop */
454
+ kv = mk_list_entry (head , struct flb_loki_kv , _head );
455
+
456
+ /* record accessor key/value pair */
457
+ if (kv -> ra_key != NULL && kv -> ra_val == NULL ) {
458
+
459
+ /* try to get the value for the record accessor */
460
+ if (flb_ra_get_kv_pair (kv -> ra_key , * map , & start_key , & out_key , & out_val )
461
+ != -1 ) {
462
+
463
+ /*
464
+ * we require the value to be a map, or it doesn't make sense as
465
+ * this is adding a map's key / values
466
+ */
467
+ if (out_val -> type != MSGPACK_OBJECT_MAP || out_val -> via .map .size <= 0 ) {
468
+ flb_plg_debug (ctx -> ins , "No valid map data found for key %s" ,
469
+ kv -> ra_key -> pattern );
470
+ }
471
+ else {
472
+ accessed_map = out_val -> via .map ;
473
+
474
+ /* for each entry in the accessed map... */
475
+ for (accessed_map_index = 0 ; accessed_map_index < accessed_map .size ;
476
+ accessed_map_index ++ ) {
477
+
478
+ /* get the entry */
479
+ accessed_map_kv = accessed_map .ptr [accessed_map_index ];
480
+
481
+ /* Pack the key and value */
482
+ flb_mp_map_header_append (mh );
483
+
484
+ pack_label_key (mp_pck , (char * ) accessed_map_kv .key .via .str .ptr ,
485
+ accessed_map_kv .key .via .str .size );
486
+
487
+ /* If the value is a string, just pack it... */
488
+ if (accessed_map_kv .val .type == MSGPACK_OBJECT_STR ) {
489
+ msgpack_pack_str_with_body (mp_pck ,
490
+ accessed_map_kv .val .via .str .ptr ,
491
+ accessed_map_kv .val .via .str .size );
492
+ }
493
+ /*
494
+ * ...otherwise convert value to JSON string, as Loki always
495
+ * requires a string value
496
+ */
497
+ else {
498
+ accessed_map_val_json = flb_msgpack_to_json_str (1024 ,
499
+ & accessed_map_kv .val );
500
+ if (accessed_map_val_json ) {
501
+ msgpack_pack_str_with_body (mp_pck , accessed_map_val_json ,
502
+ strlen (accessed_map_val_json ));
503
+ flb_free (accessed_map_val_json );
504
+ }
505
+ }
506
+ }
507
+ }
508
+ }
509
+ }
510
+ }
511
+ }
512
+
419
513
static flb_sds_t pack_structured_metadata (struct flb_loki * ctx ,
420
514
msgpack_packer * mp_pck ,
421
515
char * tag , int tag_len ,
@@ -424,7 +518,17 @@ static flb_sds_t pack_structured_metadata(struct flb_loki *ctx,
424
518
struct flb_mp_map_header mh ;
425
519
/* Initialize dynamic map header */
426
520
flb_mp_map_header_init (& mh , mp_pck );
427
- pack_kv (ctx , mp_pck , tag , tag_len , map , & mh , & ctx -> structured_metadata_list );
521
+ if (ctx -> structured_metadata_map_keys ) {
522
+ pack_maps (ctx , mp_pck , tag , tag_len , map , & mh ,
523
+ & ctx -> structured_metadata_map_keys_list );
524
+ }
525
+ /*
526
+ * explicit structured_metadata entries override
527
+ * structured_metadata_map_keys entries
528
+ * */
529
+ if (ctx -> structured_metadata ) {
530
+ pack_kv (ctx , mp_pck , tag , tag_len , map , & mh , & ctx -> structured_metadata_list );
531
+ }
428
532
flb_mp_map_header_end (& mh );
429
533
return 0 ;
430
534
}
@@ -788,6 +892,7 @@ static int parse_labels(struct flb_loki *ctx)
788
892
789
893
flb_loki_kv_init (& ctx -> labels_list );
790
894
flb_loki_kv_init (& ctx -> structured_metadata_list );
895
+ flb_loki_kv_init (& ctx -> structured_metadata_map_keys_list );
791
896
792
897
if (ctx -> structured_metadata ) {
793
898
ret = parse_kv (ctx , ctx -> structured_metadata , & ctx -> structured_metadata_list , & ra_used );
@@ -796,6 +901,28 @@ static int parse_labels(struct flb_loki *ctx)
796
901
}
797
902
}
798
903
904
+ /* Append structured metadata map keys set in the configuration */
905
+ if (ctx -> structured_metadata_map_keys ) {
906
+ mk_list_foreach (head , ctx -> structured_metadata_map_keys ) {
907
+ entry = mk_list_entry (head , struct flb_slist_entry , _head );
908
+ if (entry -> str [0 ] != '$' ) {
909
+ flb_plg_error (ctx -> ins ,
910
+ "invalid structured metadata map key, the name must start "
911
+ "with '$'" );
912
+ return -1 ;
913
+ }
914
+
915
+ ret = flb_loki_kv_append (ctx , & ctx -> structured_metadata_map_keys_list ,
916
+ entry -> str , NULL );
917
+ if (ret == -1 ) {
918
+ return -1 ;
919
+ }
920
+ else if (ret > 0 ) {
921
+ ra_used ++ ;
922
+ }
923
+ }
924
+ }
925
+
799
926
if (ctx -> labels ) {
800
927
ret = parse_kv (ctx , ctx -> labels , & ctx -> labels_list , & ra_used );
801
928
if (ret == -1 ) {
@@ -971,6 +1098,7 @@ static struct flb_loki *loki_config_create(struct flb_output_instance *ins,
971
1098
ctx -> ins = ins ;
972
1099
flb_loki_kv_init (& ctx -> labels_list );
973
1100
flb_loki_kv_init (& ctx -> structured_metadata_list );
1101
+ flb_loki_kv_init (& ctx -> structured_metadata_map_keys_list );
974
1102
975
1103
/* Register context with plugin instance */
976
1104
flb_output_set_context (ins , ctx );
@@ -1539,12 +1667,13 @@ static flb_sds_t loki_compose_payload(struct flb_loki *ctx,
1539
1667
while ((ret = flb_log_event_decoder_next (
1540
1668
& log_decoder ,
1541
1669
& log_event )) == FLB_EVENT_DECODER_SUCCESS ) {
1542
- msgpack_pack_array (& mp_pck , ctx -> structured_metadata ? 3 : 2 );
1670
+ msgpack_pack_array (& mp_pck , ctx -> structured_metadata ||
1671
+ ctx -> structured_metadata_map_keys ? 3 : 2 );
1543
1672
1544
1673
/* Append the timestamp */
1545
1674
pack_timestamp (& mp_pck , & log_event .timestamp );
1546
1675
pack_record (ctx , & mp_pck , log_event .body , dynamic_tenant_id );
1547
- if (ctx -> structured_metadata ) {
1676
+ if (ctx -> structured_metadata || ctx -> structured_metadata_map_keys ) {
1548
1677
pack_structured_metadata (ctx , & mp_pck , tag , tag_len , NULL );
1549
1678
}
1550
1679
}
@@ -1575,12 +1704,13 @@ static flb_sds_t loki_compose_payload(struct flb_loki *ctx,
1575
1704
msgpack_pack_str_body (& mp_pck , "values" , 6 );
1576
1705
msgpack_pack_array (& mp_pck , 1 );
1577
1706
1578
- msgpack_pack_array (& mp_pck , ctx -> structured_metadata ? 3 : 2 );
1707
+ msgpack_pack_array (& mp_pck , ctx -> structured_metadata ||
1708
+ ctx -> structured_metadata_map_keys ? 3 : 2 );
1579
1709
1580
1710
/* Append the timestamp */
1581
1711
pack_timestamp (& mp_pck , & log_event .timestamp );
1582
1712
pack_record (ctx , & mp_pck , log_event .body , dynamic_tenant_id );
1583
- if (ctx -> structured_metadata ) {
1713
+ if (ctx -> structured_metadata || ctx -> structured_metadata_map_keys ) {
1584
1714
pack_structured_metadata (ctx , & mp_pck , tag , tag_len , log_event .body );
1585
1715
}
1586
1716
}
@@ -1905,6 +2035,13 @@ static struct flb_config_map config_map[] = {
1905
2035
0 , FLB_TRUE , offsetof(struct flb_loki , structured_metadata ),
1906
2036
"optional structured metadata fields for API requests."
1907
2037
},
2038
+
2039
+ {
2040
+ FLB_CONFIG_MAP_CLIST , "structured_metadata_map_keys" , NULL ,
2041
+ 0 , FLB_TRUE , offsetof(struct flb_loki , structured_metadata_map_keys ),
2042
+ "optional structured metadata fields, as derived dynamically from configured maps "
2043
+ "keys, for API requests."
2044
+ },
1908
2045
1909
2046
{
1910
2047
FLB_CONFIG_MAP_BOOL , "auto_kubernetes_labels" , "false" ,
0 commit comments