rucio 32.8.6__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of rucio might be problematic. Click here for more details.

Files changed (481) hide show
  1. rucio/__init__.py +18 -0
  2. rucio/alembicrevision.py +16 -0
  3. rucio/api/__init__.py +14 -0
  4. rucio/api/account.py +266 -0
  5. rucio/api/account_limit.py +287 -0
  6. rucio/api/authentication.py +302 -0
  7. rucio/api/config.py +218 -0
  8. rucio/api/credential.py +60 -0
  9. rucio/api/did.py +726 -0
  10. rucio/api/dirac.py +71 -0
  11. rucio/api/exporter.py +60 -0
  12. rucio/api/heartbeat.py +62 -0
  13. rucio/api/identity.py +160 -0
  14. rucio/api/importer.py +46 -0
  15. rucio/api/lifetime_exception.py +95 -0
  16. rucio/api/lock.py +131 -0
  17. rucio/api/meta.py +85 -0
  18. rucio/api/permission.py +72 -0
  19. rucio/api/quarantined_replica.py +69 -0
  20. rucio/api/replica.py +528 -0
  21. rucio/api/request.py +220 -0
  22. rucio/api/rse.py +601 -0
  23. rucio/api/rule.py +335 -0
  24. rucio/api/scope.py +89 -0
  25. rucio/api/subscription.py +255 -0
  26. rucio/api/temporary_did.py +49 -0
  27. rucio/api/vo.py +112 -0
  28. rucio/client/__init__.py +16 -0
  29. rucio/client/accountclient.py +413 -0
  30. rucio/client/accountlimitclient.py +155 -0
  31. rucio/client/baseclient.py +929 -0
  32. rucio/client/client.py +77 -0
  33. rucio/client/configclient.py +113 -0
  34. rucio/client/credentialclient.py +54 -0
  35. rucio/client/didclient.py +691 -0
  36. rucio/client/diracclient.py +48 -0
  37. rucio/client/downloadclient.py +1674 -0
  38. rucio/client/exportclient.py +44 -0
  39. rucio/client/fileclient.py +51 -0
  40. rucio/client/importclient.py +42 -0
  41. rucio/client/lifetimeclient.py +74 -0
  42. rucio/client/lockclient.py +99 -0
  43. rucio/client/metaclient.py +137 -0
  44. rucio/client/pingclient.py +45 -0
  45. rucio/client/replicaclient.py +444 -0
  46. rucio/client/requestclient.py +109 -0
  47. rucio/client/rseclient.py +664 -0
  48. rucio/client/ruleclient.py +287 -0
  49. rucio/client/scopeclient.py +88 -0
  50. rucio/client/subscriptionclient.py +161 -0
  51. rucio/client/touchclient.py +78 -0
  52. rucio/client/uploadclient.py +871 -0
  53. rucio/common/__init__.py +14 -0
  54. rucio/common/cache.py +74 -0
  55. rucio/common/config.py +796 -0
  56. rucio/common/constants.py +92 -0
  57. rucio/common/constraints.py +18 -0
  58. rucio/common/didtype.py +187 -0
  59. rucio/common/dumper/__init__.py +306 -0
  60. rucio/common/dumper/consistency.py +449 -0
  61. rucio/common/dumper/data_models.py +325 -0
  62. rucio/common/dumper/path_parsing.py +65 -0
  63. rucio/common/exception.py +1092 -0
  64. rucio/common/extra.py +37 -0
  65. rucio/common/logging.py +404 -0
  66. rucio/common/pcache.py +1387 -0
  67. rucio/common/policy.py +84 -0
  68. rucio/common/schema/__init__.py +143 -0
  69. rucio/common/schema/atlas.py +411 -0
  70. rucio/common/schema/belleii.py +406 -0
  71. rucio/common/schema/cms.py +478 -0
  72. rucio/common/schema/domatpc.py +399 -0
  73. rucio/common/schema/escape.py +424 -0
  74. rucio/common/schema/generic.py +431 -0
  75. rucio/common/schema/generic_multi_vo.py +410 -0
  76. rucio/common/schema/icecube.py +404 -0
  77. rucio/common/schema/lsst.py +423 -0
  78. rucio/common/stomp_utils.py +160 -0
  79. rucio/common/stopwatch.py +56 -0
  80. rucio/common/test_rucio_server.py +148 -0
  81. rucio/common/types.py +158 -0
  82. rucio/common/utils.py +1946 -0
  83. rucio/core/__init__.py +14 -0
  84. rucio/core/account.py +426 -0
  85. rucio/core/account_counter.py +171 -0
  86. rucio/core/account_limit.py +357 -0
  87. rucio/core/authentication.py +563 -0
  88. rucio/core/config.py +386 -0
  89. rucio/core/credential.py +218 -0
  90. rucio/core/did.py +3102 -0
  91. rucio/core/did_meta_plugins/__init__.py +250 -0
  92. rucio/core/did_meta_plugins/did_column_meta.py +326 -0
  93. rucio/core/did_meta_plugins/did_meta_plugin_interface.py +116 -0
  94. rucio/core/did_meta_plugins/filter_engine.py +573 -0
  95. rucio/core/did_meta_plugins/json_meta.py +215 -0
  96. rucio/core/did_meta_plugins/mongo_meta.py +199 -0
  97. rucio/core/did_meta_plugins/postgres_meta.py +317 -0
  98. rucio/core/dirac.py +208 -0
  99. rucio/core/distance.py +164 -0
  100. rucio/core/exporter.py +59 -0
  101. rucio/core/heartbeat.py +263 -0
  102. rucio/core/identity.py +290 -0
  103. rucio/core/importer.py +248 -0
  104. rucio/core/lifetime_exception.py +377 -0
  105. rucio/core/lock.py +474 -0
  106. rucio/core/message.py +241 -0
  107. rucio/core/meta.py +190 -0
  108. rucio/core/monitor.py +441 -0
  109. rucio/core/naming_convention.py +154 -0
  110. rucio/core/nongrid_trace.py +124 -0
  111. rucio/core/oidc.py +1339 -0
  112. rucio/core/permission/__init__.py +107 -0
  113. rucio/core/permission/atlas.py +1333 -0
  114. rucio/core/permission/belleii.py +1076 -0
  115. rucio/core/permission/cms.py +1166 -0
  116. rucio/core/permission/escape.py +1076 -0
  117. rucio/core/permission/generic.py +1128 -0
  118. rucio/core/permission/generic_multi_vo.py +1148 -0
  119. rucio/core/quarantined_replica.py +190 -0
  120. rucio/core/replica.py +3627 -0
  121. rucio/core/replica_sorter.py +368 -0
  122. rucio/core/request.py +2241 -0
  123. rucio/core/rse.py +1835 -0
  124. rucio/core/rse_counter.py +155 -0
  125. rucio/core/rse_expression_parser.py +460 -0
  126. rucio/core/rse_selector.py +277 -0
  127. rucio/core/rule.py +3419 -0
  128. rucio/core/rule_grouping.py +1473 -0
  129. rucio/core/scope.py +152 -0
  130. rucio/core/subscription.py +316 -0
  131. rucio/core/temporary_did.py +188 -0
  132. rucio/core/topology.py +448 -0
  133. rucio/core/trace.py +361 -0
  134. rucio/core/transfer.py +1233 -0
  135. rucio/core/vo.py +151 -0
  136. rucio/core/volatile_replica.py +123 -0
  137. rucio/daemons/__init__.py +14 -0
  138. rucio/daemons/abacus/__init__.py +14 -0
  139. rucio/daemons/abacus/account.py +106 -0
  140. rucio/daemons/abacus/collection_replica.py +113 -0
  141. rucio/daemons/abacus/rse.py +107 -0
  142. rucio/daemons/atropos/__init__.py +14 -0
  143. rucio/daemons/atropos/atropos.py +243 -0
  144. rucio/daemons/auditor/__init__.py +261 -0
  145. rucio/daemons/auditor/hdfs.py +86 -0
  146. rucio/daemons/auditor/srmdumps.py +284 -0
  147. rucio/daemons/automatix/__init__.py +14 -0
  148. rucio/daemons/automatix/automatix.py +281 -0
  149. rucio/daemons/badreplicas/__init__.py +14 -0
  150. rucio/daemons/badreplicas/minos.py +311 -0
  151. rucio/daemons/badreplicas/minos_temporary_expiration.py +173 -0
  152. rucio/daemons/badreplicas/necromancer.py +200 -0
  153. rucio/daemons/bb8/__init__.py +14 -0
  154. rucio/daemons/bb8/bb8.py +356 -0
  155. rucio/daemons/bb8/common.py +762 -0
  156. rucio/daemons/bb8/nuclei_background_rebalance.py +147 -0
  157. rucio/daemons/bb8/t2_background_rebalance.py +146 -0
  158. rucio/daemons/c3po/__init__.py +14 -0
  159. rucio/daemons/c3po/algorithms/__init__.py +14 -0
  160. rucio/daemons/c3po/algorithms/simple.py +131 -0
  161. rucio/daemons/c3po/algorithms/t2_free_space.py +125 -0
  162. rucio/daemons/c3po/algorithms/t2_free_space_only_pop.py +127 -0
  163. rucio/daemons/c3po/algorithms/t2_free_space_only_pop_with_network.py +279 -0
  164. rucio/daemons/c3po/c3po.py +342 -0
  165. rucio/daemons/c3po/collectors/__init__.py +14 -0
  166. rucio/daemons/c3po/collectors/agis.py +108 -0
  167. rucio/daemons/c3po/collectors/free_space.py +62 -0
  168. rucio/daemons/c3po/collectors/jedi_did.py +48 -0
  169. rucio/daemons/c3po/collectors/mock_did.py +46 -0
  170. rucio/daemons/c3po/collectors/network_metrics.py +63 -0
  171. rucio/daemons/c3po/collectors/workload.py +110 -0
  172. rucio/daemons/c3po/utils/__init__.py +14 -0
  173. rucio/daemons/c3po/utils/dataset_cache.py +40 -0
  174. rucio/daemons/c3po/utils/expiring_dataset_cache.py +45 -0
  175. rucio/daemons/c3po/utils/expiring_list.py +63 -0
  176. rucio/daemons/c3po/utils/popularity.py +82 -0
  177. rucio/daemons/c3po/utils/timeseries.py +76 -0
  178. rucio/daemons/cache/__init__.py +14 -0
  179. rucio/daemons/cache/consumer.py +191 -0
  180. rucio/daemons/common.py +391 -0
  181. rucio/daemons/conveyor/__init__.py +14 -0
  182. rucio/daemons/conveyor/common.py +530 -0
  183. rucio/daemons/conveyor/finisher.py +492 -0
  184. rucio/daemons/conveyor/poller.py +372 -0
  185. rucio/daemons/conveyor/preparer.py +198 -0
  186. rucio/daemons/conveyor/receiver.py +206 -0
  187. rucio/daemons/conveyor/stager.py +127 -0
  188. rucio/daemons/conveyor/submitter.py +379 -0
  189. rucio/daemons/conveyor/throttler.py +468 -0
  190. rucio/daemons/follower/__init__.py +14 -0
  191. rucio/daemons/follower/follower.py +97 -0
  192. rucio/daemons/hermes/__init__.py +14 -0
  193. rucio/daemons/hermes/hermes.py +738 -0
  194. rucio/daemons/judge/__init__.py +14 -0
  195. rucio/daemons/judge/cleaner.py +149 -0
  196. rucio/daemons/judge/evaluator.py +172 -0
  197. rucio/daemons/judge/injector.py +154 -0
  198. rucio/daemons/judge/repairer.py +144 -0
  199. rucio/daemons/oauthmanager/__init__.py +14 -0
  200. rucio/daemons/oauthmanager/oauthmanager.py +199 -0
  201. rucio/daemons/reaper/__init__.py +14 -0
  202. rucio/daemons/reaper/dark_reaper.py +272 -0
  203. rucio/daemons/reaper/light_reaper.py +255 -0
  204. rucio/daemons/reaper/reaper.py +701 -0
  205. rucio/daemons/replicarecoverer/__init__.py +14 -0
  206. rucio/daemons/replicarecoverer/suspicious_replica_recoverer.py +487 -0
  207. rucio/daemons/storage/__init__.py +14 -0
  208. rucio/daemons/storage/consistency/__init__.py +14 -0
  209. rucio/daemons/storage/consistency/actions.py +753 -0
  210. rucio/daemons/tracer/__init__.py +14 -0
  211. rucio/daemons/tracer/kronos.py +513 -0
  212. rucio/daemons/transmogrifier/__init__.py +14 -0
  213. rucio/daemons/transmogrifier/transmogrifier.py +753 -0
  214. rucio/daemons/undertaker/__init__.py +14 -0
  215. rucio/daemons/undertaker/undertaker.py +137 -0
  216. rucio/db/__init__.py +14 -0
  217. rucio/db/sqla/__init__.py +38 -0
  218. rucio/db/sqla/constants.py +192 -0
  219. rucio/db/sqla/migrate_repo/__init__.py +14 -0
  220. rucio/db/sqla/migrate_repo/env.py +111 -0
  221. rucio/db/sqla/migrate_repo/versions/01eaf73ab656_add_new_rule_notification_state_progress.py +71 -0
  222. rucio/db/sqla/migrate_repo/versions/0437a40dbfd1_add_eol_at_in_rules.py +50 -0
  223. rucio/db/sqla/migrate_repo/versions/0f1adb7a599a_create_transfer_hops_table.py +61 -0
  224. rucio/db/sqla/migrate_repo/versions/102efcf145f4_added_stuck_at_column_to_rules.py +46 -0
  225. rucio/db/sqla/migrate_repo/versions/13d4f70c66a9_introduce_transfer_limits.py +93 -0
  226. rucio/db/sqla/migrate_repo/versions/140fef722e91_cleanup_distances_table.py +78 -0
  227. rucio/db/sqla/migrate_repo/versions/14ec5aeb64cf_add_request_external_host.py +46 -0
  228. rucio/db/sqla/migrate_repo/versions/156fb5b5a14_add_request_type_to_requests_idx.py +53 -0
  229. rucio/db/sqla/migrate_repo/versions/1677d4d803c8_split_rse_availability_into_multiple.py +69 -0
  230. rucio/db/sqla/migrate_repo/versions/16a0aca82e12_create_index_on_table_replicas_path.py +42 -0
  231. rucio/db/sqla/migrate_repo/versions/1803333ac20f_adding_provenance_and_phys_group.py +46 -0
  232. rucio/db/sqla/migrate_repo/versions/1a29d6a9504c_add_didtype_chck_to_requests.py +61 -0
  233. rucio/db/sqla/migrate_repo/versions/1a80adff031a_create_index_on_rules_hist_recent.py +42 -0
  234. rucio/db/sqla/migrate_repo/versions/1c45d9730ca6_increase_identity_length.py +141 -0
  235. rucio/db/sqla/migrate_repo/versions/1d1215494e95_add_quarantined_replicas_table.py +75 -0
  236. rucio/db/sqla/migrate_repo/versions/1d96f484df21_asynchronous_rules_and_rule_approval.py +75 -0
  237. rucio/db/sqla/migrate_repo/versions/1f46c5f240ac_add_bytes_column_to_bad_replicas.py +46 -0
  238. rucio/db/sqla/migrate_repo/versions/1fc15ab60d43_add_message_history_table.py +51 -0
  239. rucio/db/sqla/migrate_repo/versions/2190e703eb6e_move_rse_settings_to_rse_attributes.py +135 -0
  240. rucio/db/sqla/migrate_repo/versions/21d6b9dc9961_add_mismatch_scheme_state_to_requests.py +65 -0
  241. rucio/db/sqla/migrate_repo/versions/22cf51430c78_add_availability_column_to_table_rses.py +42 -0
  242. rucio/db/sqla/migrate_repo/versions/22d887e4ec0a_create_sources_table.py +66 -0
  243. rucio/db/sqla/migrate_repo/versions/25821a8a45a3_remove_unique_constraint_on_requests.py +54 -0
  244. rucio/db/sqla/migrate_repo/versions/25fc855625cf_added_unique_constraint_to_rules.py +43 -0
  245. rucio/db/sqla/migrate_repo/versions/269fee20dee9_add_repair_cnt_to_locks.py +46 -0
  246. rucio/db/sqla/migrate_repo/versions/271a46ea6244_add_ignore_availability_column_to_rules.py +47 -0
  247. rucio/db/sqla/migrate_repo/versions/277b5fbb41d3_switch_heartbeats_executable.py +54 -0
  248. rucio/db/sqla/migrate_repo/versions/27e3a68927fb_remove_replicas_tombstone_and_replicas_.py +39 -0
  249. rucio/db/sqla/migrate_repo/versions/2854cd9e168_added_rule_id_column.py +48 -0
  250. rucio/db/sqla/migrate_repo/versions/295289b5a800_processed_by_and__at_in_requests.py +47 -0
  251. rucio/db/sqla/migrate_repo/versions/2962ece31cf4_add_nbaccesses_column_in_the_did_table.py +48 -0
  252. rucio/db/sqla/migrate_repo/versions/2af3291ec4c_added_replicas_history_table.py +59 -0
  253. rucio/db/sqla/migrate_repo/versions/2b69addda658_add_columns_for_third_party_copy_read_.py +47 -0
  254. rucio/db/sqla/migrate_repo/versions/2b8e7bcb4783_add_config_table.py +72 -0
  255. rucio/db/sqla/migrate_repo/versions/2ba5229cb54c_add_submitted_at_to_requests_table.py +46 -0
  256. rucio/db/sqla/migrate_repo/versions/2cbee484dcf9_added_column_volume_to_rse_transfer_.py +45 -0
  257. rucio/db/sqla/migrate_repo/versions/2edee4a83846_add_source_to_requests_and_requests_.py +48 -0
  258. rucio/db/sqla/migrate_repo/versions/2eef46be23d4_change_tokens_pk.py +48 -0
  259. rucio/db/sqla/migrate_repo/versions/2f648fc909f3_index_in_rule_history_on_scope_name.py +42 -0
  260. rucio/db/sqla/migrate_repo/versions/3082b8cef557_add_naming_convention_table_and_closed_.py +69 -0
  261. rucio/db/sqla/migrate_repo/versions/30fa38b6434e_add_index_on_service_column_in_the_message_table.py +46 -0
  262. rucio/db/sqla/migrate_repo/versions/3152492b110b_added_staging_area_column.py +78 -0
  263. rucio/db/sqla/migrate_repo/versions/32c7d2783f7e_create_bad_replicas_table.py +62 -0
  264. rucio/db/sqla/migrate_repo/versions/3345511706b8_replicas_table_pk_definition_is_in_.py +74 -0
  265. rucio/db/sqla/migrate_repo/versions/35ef10d1e11b_change_index_on_table_requests.py +44 -0
  266. rucio/db/sqla/migrate_repo/versions/379a19b5332d_create_rse_limits_table.py +67 -0
  267. rucio/db/sqla/migrate_repo/versions/384b96aa0f60_created_rule_history_tables.py +134 -0
  268. rucio/db/sqla/migrate_repo/versions/3ac1660a1a72_extend_distance_table.py +58 -0
  269. rucio/db/sqla/migrate_repo/versions/3ad36e2268b0_create_collection_replicas_updates_table.py +79 -0
  270. rucio/db/sqla/migrate_repo/versions/3c9df354071b_extend_waiting_request_state.py +61 -0
  271. rucio/db/sqla/migrate_repo/versions/3d9813fab443_add_a_new_state_lost_in_badfilesstatus.py +45 -0
  272. rucio/db/sqla/migrate_repo/versions/40ad39ce3160_add_transferred_at_to_requests_table.py +46 -0
  273. rucio/db/sqla/migrate_repo/versions/4207be2fd914_add_notification_column_to_rules.py +65 -0
  274. rucio/db/sqla/migrate_repo/versions/42db2617c364_create_index_on_requests_external_id.py +42 -0
  275. rucio/db/sqla/migrate_repo/versions/436827b13f82_added_column_activity_to_table_requests.py +46 -0
  276. rucio/db/sqla/migrate_repo/versions/44278720f774_update_requests_typ_sta_upd_idx_index.py +46 -0
  277. rucio/db/sqla/migrate_repo/versions/45378a1e76a8_create_collection_replica_table.py +80 -0
  278. rucio/db/sqla/migrate_repo/versions/469d262be19_removing_created_at_index.py +43 -0
  279. rucio/db/sqla/migrate_repo/versions/4783c1f49cb4_create_distance_table.py +61 -0
  280. rucio/db/sqla/migrate_repo/versions/49a21b4d4357_create_index_on_table_tokens.py +47 -0
  281. rucio/db/sqla/migrate_repo/versions/4a2cbedda8b9_add_source_replica_expression_column_to_.py +46 -0
  282. rucio/db/sqla/migrate_repo/versions/4a7182d9578b_added_bytes_length_accessed_at_columns.py +52 -0
  283. rucio/db/sqla/migrate_repo/versions/4bab9edd01fc_create_index_on_requests_rule_id.py +42 -0
  284. rucio/db/sqla/migrate_repo/versions/4c3a4acfe006_new_attr_account_table.py +65 -0
  285. rucio/db/sqla/migrate_repo/versions/4cf0a2e127d4_adding_transient_metadata.py +46 -0
  286. rucio/db/sqla/migrate_repo/versions/50280c53117c_add_qos_class_to_rse.py +47 -0
  287. rucio/db/sqla/migrate_repo/versions/52153819589c_add_rse_id_to_replicas_table.py +45 -0
  288. rucio/db/sqla/migrate_repo/versions/52fd9f4916fa_added_activity_to_rules.py +46 -0
  289. rucio/db/sqla/migrate_repo/versions/53b479c3cb0f_fix_did_meta_table_missing_updated_at_.py +48 -0
  290. rucio/db/sqla/migrate_repo/versions/5673b4b6e843_add_wfms_metadata_to_rule_tables.py +50 -0
  291. rucio/db/sqla/migrate_repo/versions/575767d9f89_added_source_history_table.py +59 -0
  292. rucio/db/sqla/migrate_repo/versions/58bff7008037_add_started_at_to_requests.py +48 -0
  293. rucio/db/sqla/migrate_repo/versions/58c8b78301ab_rename_callback_to_message.py +108 -0
  294. rucio/db/sqla/migrate_repo/versions/5f139f77382a_added_child_rule_id_column.py +57 -0
  295. rucio/db/sqla/migrate_repo/versions/688ef1840840_adding_did_meta_table.py +51 -0
  296. rucio/db/sqla/migrate_repo/versions/6e572a9bfbf3_add_new_split_container_column_to_rules.py +50 -0
  297. rucio/db/sqla/migrate_repo/versions/70587619328_add_comment_column_for_subscriptions.py +46 -0
  298. rucio/db/sqla/migrate_repo/versions/739064d31565_remove_history_table_pks.py +42 -0
  299. rucio/db/sqla/migrate_repo/versions/7541902bf173_add_didsfollowed_and_followevents_table.py +93 -0
  300. rucio/db/sqla/migrate_repo/versions/7ec22226cdbf_new_replica_state_for_temporary_.py +73 -0
  301. rucio/db/sqla/migrate_repo/versions/810a41685bc1_added_columns_rse_transfer_limits.py +52 -0
  302. rucio/db/sqla/migrate_repo/versions/83f991c63a93_correct_rse_expression_length.py +45 -0
  303. rucio/db/sqla/migrate_repo/versions/8523998e2e76_increase_size_of_extended_attributes_.py +46 -0
  304. rucio/db/sqla/migrate_repo/versions/8ea9122275b1_adding_missing_function_based_indices.py +54 -0
  305. rucio/db/sqla/migrate_repo/versions/90f47792bb76_add_clob_payload_to_messages.py +48 -0
  306. rucio/db/sqla/migrate_repo/versions/914b8f02df38_new_table_for_lifetime_model_exceptions.py +70 -0
  307. rucio/db/sqla/migrate_repo/versions/94a5961ddbf2_add_estimator_columns.py +48 -0
  308. rucio/db/sqla/migrate_repo/versions/9a1b149a2044_add_saml_identity_type.py +95 -0
  309. rucio/db/sqla/migrate_repo/versions/9a45bc4ea66d_add_vp_table.py +55 -0
  310. rucio/db/sqla/migrate_repo/versions/9eb936a81eb1_true_is_true.py +74 -0
  311. rucio/db/sqla/migrate_repo/versions/a118956323f8_added_vo_table_and_vo_col_to_rse.py +78 -0
  312. rucio/db/sqla/migrate_repo/versions/a193a275255c_add_status_column_in_messages.py +49 -0
  313. rucio/db/sqla/migrate_repo/versions/a5f6f6e928a7_1_7_0.py +124 -0
  314. rucio/db/sqla/migrate_repo/versions/a616581ee47_added_columns_to_table_requests.py +60 -0
  315. rucio/db/sqla/migrate_repo/versions/a6eb23955c28_state_idx_non_functional.py +53 -0
  316. rucio/db/sqla/migrate_repo/versions/a74275a1ad30_added_global_quota_table.py +56 -0
  317. rucio/db/sqla/migrate_repo/versions/a93e4e47bda_heartbeats.py +67 -0
  318. rucio/db/sqla/migrate_repo/versions/ae2a56fcc89_added_comment_column_to_rules.py +50 -0
  319. rucio/db/sqla/migrate_repo/versions/b4293a99f344_added_column_identity_to_table_tokens.py +46 -0
  320. rucio/db/sqla/migrate_repo/versions/b7d287de34fd_removal_of_replicastate_source.py +92 -0
  321. rucio/db/sqla/migrate_repo/versions/b818052fa670_add_index_to_quarantined_replicas.py +42 -0
  322. rucio/db/sqla/migrate_repo/versions/b8caac94d7f0_add_comments_column_for_subscriptions_.py +46 -0
  323. rucio/db/sqla/migrate_repo/versions/b96a1c7e1cc4_new_bad_pfns_table_and_bad_replicas_.py +147 -0
  324. rucio/db/sqla/migrate_repo/versions/bb695f45c04_extend_request_state.py +78 -0
  325. rucio/db/sqla/migrate_repo/versions/bc68e9946deb_add_staging_timestamps_to_request.py +53 -0
  326. rucio/db/sqla/migrate_repo/versions/bf3baa1c1474_correct_pk_and_idx_for_history_tables.py +74 -0
  327. rucio/db/sqla/migrate_repo/versions/c0937668555f_add_qos_policy_map_table.py +56 -0
  328. rucio/db/sqla/migrate_repo/versions/c129ccdb2d5_add_lumiblocknr_to_dids.py +46 -0
  329. rucio/db/sqla/migrate_repo/versions/ccdbcd48206e_add_did_type_column_index_on_did_meta_.py +68 -0
  330. rucio/db/sqla/migrate_repo/versions/cebad904c4dd_new_payload_column_for_heartbeats.py +48 -0
  331. rucio/db/sqla/migrate_repo/versions/d1189a09c6e0_oauth2_0_and_jwt_feature_support_adding_.py +149 -0
  332. rucio/db/sqla/migrate_repo/versions/d23453595260_extend_request_state_for_preparer.py +106 -0
  333. rucio/db/sqla/migrate_repo/versions/d6dceb1de2d_added_purge_column_to_rules.py +47 -0
  334. rucio/db/sqla/migrate_repo/versions/d6e2c3b2cf26_remove_third_party_copy_column_from_rse.py +45 -0
  335. rucio/db/sqla/migrate_repo/versions/d91002c5841_new_account_limits_table.py +105 -0
  336. rucio/db/sqla/migrate_repo/versions/e138c364ebd0_extending_columns_for_filter_and_.py +52 -0
  337. rucio/db/sqla/migrate_repo/versions/e59300c8b179_support_for_archive.py +106 -0
  338. rucio/db/sqla/migrate_repo/versions/f1b14a8c2ac1_postgres_use_check_constraints.py +30 -0
  339. rucio/db/sqla/migrate_repo/versions/f41ffe206f37_oracle_global_temporary_tables.py +75 -0
  340. rucio/db/sqla/migrate_repo/versions/f85a2962b021_adding_transfertool_column_to_requests_.py +49 -0
  341. rucio/db/sqla/migrate_repo/versions/fa7a7d78b602_increase_refresh_token_size.py +45 -0
  342. rucio/db/sqla/migrate_repo/versions/fb28a95fe288_add_replicas_rse_id_tombstone_idx.py +38 -0
  343. rucio/db/sqla/migrate_repo/versions/fe1a65b176c9_set_third_party_copy_read_and_write_.py +44 -0
  344. rucio/db/sqla/migrate_repo/versions/fe8ea2fa9788_added_third_party_copy_column_to_rse_.py +46 -0
  345. rucio/db/sqla/models.py +1834 -0
  346. rucio/db/sqla/sautils.py +48 -0
  347. rucio/db/sqla/session.py +470 -0
  348. rucio/db/sqla/types.py +207 -0
  349. rucio/db/sqla/util.py +521 -0
  350. rucio/rse/__init__.py +97 -0
  351. rucio/rse/protocols/__init__.py +14 -0
  352. rucio/rse/protocols/cache.py +123 -0
  353. rucio/rse/protocols/dummy.py +112 -0
  354. rucio/rse/protocols/gfal.py +701 -0
  355. rucio/rse/protocols/globus.py +243 -0
  356. rucio/rse/protocols/gsiftp.py +93 -0
  357. rucio/rse/protocols/http_cache.py +83 -0
  358. rucio/rse/protocols/mock.py +124 -0
  359. rucio/rse/protocols/ngarc.py +210 -0
  360. rucio/rse/protocols/posix.py +251 -0
  361. rucio/rse/protocols/protocol.py +530 -0
  362. rucio/rse/protocols/rclone.py +365 -0
  363. rucio/rse/protocols/rfio.py +137 -0
  364. rucio/rse/protocols/srm.py +339 -0
  365. rucio/rse/protocols/ssh.py +414 -0
  366. rucio/rse/protocols/storm.py +207 -0
  367. rucio/rse/protocols/webdav.py +547 -0
  368. rucio/rse/protocols/xrootd.py +295 -0
  369. rucio/rse/rsemanager.py +752 -0
  370. rucio/tests/__init__.py +14 -0
  371. rucio/tests/common.py +244 -0
  372. rucio/tests/common_server.py +132 -0
  373. rucio/transfertool/__init__.py +14 -0
  374. rucio/transfertool/fts3.py +1484 -0
  375. rucio/transfertool/globus.py +200 -0
  376. rucio/transfertool/globus_library.py +182 -0
  377. rucio/transfertool/mock.py +81 -0
  378. rucio/transfertool/transfertool.py +212 -0
  379. rucio/vcsversion.py +11 -0
  380. rucio/version.py +46 -0
  381. rucio/web/__init__.py +14 -0
  382. rucio/web/rest/__init__.py +14 -0
  383. rucio/web/rest/flaskapi/__init__.py +14 -0
  384. rucio/web/rest/flaskapi/authenticated_bp.py +28 -0
  385. rucio/web/rest/flaskapi/v1/__init__.py +14 -0
  386. rucio/web/rest/flaskapi/v1/accountlimits.py +234 -0
  387. rucio/web/rest/flaskapi/v1/accounts.py +1088 -0
  388. rucio/web/rest/flaskapi/v1/archives.py +100 -0
  389. rucio/web/rest/flaskapi/v1/auth.py +1642 -0
  390. rucio/web/rest/flaskapi/v1/common.py +385 -0
  391. rucio/web/rest/flaskapi/v1/config.py +305 -0
  392. rucio/web/rest/flaskapi/v1/credentials.py +213 -0
  393. rucio/web/rest/flaskapi/v1/dids.py +2204 -0
  394. rucio/web/rest/flaskapi/v1/dirac.py +116 -0
  395. rucio/web/rest/flaskapi/v1/export.py +77 -0
  396. rucio/web/rest/flaskapi/v1/heartbeats.py +129 -0
  397. rucio/web/rest/flaskapi/v1/identities.py +263 -0
  398. rucio/web/rest/flaskapi/v1/import.py +133 -0
  399. rucio/web/rest/flaskapi/v1/lifetime_exceptions.py +315 -0
  400. rucio/web/rest/flaskapi/v1/locks.py +360 -0
  401. rucio/web/rest/flaskapi/v1/main.py +83 -0
  402. rucio/web/rest/flaskapi/v1/meta.py +226 -0
  403. rucio/web/rest/flaskapi/v1/metrics.py +37 -0
  404. rucio/web/rest/flaskapi/v1/nongrid_traces.py +97 -0
  405. rucio/web/rest/flaskapi/v1/ping.py +89 -0
  406. rucio/web/rest/flaskapi/v1/redirect.py +366 -0
  407. rucio/web/rest/flaskapi/v1/replicas.py +1866 -0
  408. rucio/web/rest/flaskapi/v1/requests.py +841 -0
  409. rucio/web/rest/flaskapi/v1/rses.py +2204 -0
  410. rucio/web/rest/flaskapi/v1/rules.py +824 -0
  411. rucio/web/rest/flaskapi/v1/scopes.py +161 -0
  412. rucio/web/rest/flaskapi/v1/subscriptions.py +646 -0
  413. rucio/web/rest/flaskapi/v1/templates/auth_crash.html +80 -0
  414. rucio/web/rest/flaskapi/v1/templates/auth_granted.html +82 -0
  415. rucio/web/rest/flaskapi/v1/tmp_dids.py +115 -0
  416. rucio/web/rest/flaskapi/v1/traces.py +100 -0
  417. rucio/web/rest/flaskapi/v1/vos.py +280 -0
  418. rucio/web/rest/main.py +19 -0
  419. rucio/web/rest/metrics.py +28 -0
  420. rucio-32.8.6.data/data/rucio/etc/alembic.ini.template +71 -0
  421. rucio-32.8.6.data/data/rucio/etc/alembic_offline.ini.template +74 -0
  422. rucio-32.8.6.data/data/rucio/etc/globus-config.yml.template +5 -0
  423. rucio-32.8.6.data/data/rucio/etc/ldap.cfg.template +30 -0
  424. rucio-32.8.6.data/data/rucio/etc/mail_templates/rule_approval_request.tmpl +38 -0
  425. rucio-32.8.6.data/data/rucio/etc/mail_templates/rule_approved_admin.tmpl +4 -0
  426. rucio-32.8.6.data/data/rucio/etc/mail_templates/rule_approved_user.tmpl +17 -0
  427. rucio-32.8.6.data/data/rucio/etc/mail_templates/rule_denied_admin.tmpl +6 -0
  428. rucio-32.8.6.data/data/rucio/etc/mail_templates/rule_denied_user.tmpl +17 -0
  429. rucio-32.8.6.data/data/rucio/etc/mail_templates/rule_ok_notification.tmpl +19 -0
  430. rucio-32.8.6.data/data/rucio/etc/rse-accounts.cfg.template +25 -0
  431. rucio-32.8.6.data/data/rucio/etc/rucio.cfg.atlas.client.template +42 -0
  432. rucio-32.8.6.data/data/rucio/etc/rucio.cfg.template +257 -0
  433. rucio-32.8.6.data/data/rucio/etc/rucio_multi_vo.cfg.template +234 -0
  434. rucio-32.8.6.data/data/rucio/requirements.txt +55 -0
  435. rucio-32.8.6.data/data/rucio/tools/bootstrap.py +34 -0
  436. rucio-32.8.6.data/data/rucio/tools/merge_rucio_configs.py +147 -0
  437. rucio-32.8.6.data/data/rucio/tools/reset_database.py +40 -0
  438. rucio-32.8.6.data/scripts/rucio +2540 -0
  439. rucio-32.8.6.data/scripts/rucio-abacus-account +75 -0
  440. rucio-32.8.6.data/scripts/rucio-abacus-collection-replica +47 -0
  441. rucio-32.8.6.data/scripts/rucio-abacus-rse +79 -0
  442. rucio-32.8.6.data/scripts/rucio-admin +2434 -0
  443. rucio-32.8.6.data/scripts/rucio-atropos +61 -0
  444. rucio-32.8.6.data/scripts/rucio-auditor +199 -0
  445. rucio-32.8.6.data/scripts/rucio-automatix +51 -0
  446. rucio-32.8.6.data/scripts/rucio-bb8 +58 -0
  447. rucio-32.8.6.data/scripts/rucio-c3po +86 -0
  448. rucio-32.8.6.data/scripts/rucio-cache-client +135 -0
  449. rucio-32.8.6.data/scripts/rucio-cache-consumer +43 -0
  450. rucio-32.8.6.data/scripts/rucio-conveyor-finisher +59 -0
  451. rucio-32.8.6.data/scripts/rucio-conveyor-poller +67 -0
  452. rucio-32.8.6.data/scripts/rucio-conveyor-preparer +38 -0
  453. rucio-32.8.6.data/scripts/rucio-conveyor-receiver +44 -0
  454. rucio-32.8.6.data/scripts/rucio-conveyor-stager +77 -0
  455. rucio-32.8.6.data/scripts/rucio-conveyor-submitter +140 -0
  456. rucio-32.8.6.data/scripts/rucio-conveyor-throttler +105 -0
  457. rucio-32.8.6.data/scripts/rucio-dark-reaper +54 -0
  458. rucio-32.8.6.data/scripts/rucio-dumper +159 -0
  459. rucio-32.8.6.data/scripts/rucio-follower +45 -0
  460. rucio-32.8.6.data/scripts/rucio-hermes +55 -0
  461. rucio-32.8.6.data/scripts/rucio-judge-cleaner +90 -0
  462. rucio-32.8.6.data/scripts/rucio-judge-evaluator +138 -0
  463. rucio-32.8.6.data/scripts/rucio-judge-injector +45 -0
  464. rucio-32.8.6.data/scripts/rucio-judge-repairer +45 -0
  465. rucio-32.8.6.data/scripts/rucio-kronos +45 -0
  466. rucio-32.8.6.data/scripts/rucio-light-reaper +53 -0
  467. rucio-32.8.6.data/scripts/rucio-minos +54 -0
  468. rucio-32.8.6.data/scripts/rucio-minos-temporary-expiration +51 -0
  469. rucio-32.8.6.data/scripts/rucio-necromancer +121 -0
  470. rucio-32.8.6.data/scripts/rucio-oauth-manager +64 -0
  471. rucio-32.8.6.data/scripts/rucio-reaper +84 -0
  472. rucio-32.8.6.data/scripts/rucio-replica-recoverer +249 -0
  473. rucio-32.8.6.data/scripts/rucio-storage-consistency-actions +75 -0
  474. rucio-32.8.6.data/scripts/rucio-transmogrifier +78 -0
  475. rucio-32.8.6.data/scripts/rucio-undertaker +77 -0
  476. rucio-32.8.6.dist-info/METADATA +83 -0
  477. rucio-32.8.6.dist-info/RECORD +481 -0
  478. rucio-32.8.6.dist-info/WHEEL +5 -0
  479. rucio-32.8.6.dist-info/licenses/AUTHORS.rst +94 -0
  480. rucio-32.8.6.dist-info/licenses/LICENSE +201 -0
  481. rucio-32.8.6.dist-info/top_level.txt +1 -0
@@ -0,0 +1,738 @@
1
+ # -*- coding: utf-8 -*-
2
+ # Copyright European Organization for Nuclear Research (CERN) since 2012
3
+ #
4
+ # Licensed under the Apache License, Version 2.0 (the "License");
5
+ # you may not use this file except in compliance with the License.
6
+ # You may obtain a copy of the License at
7
+ #
8
+ # http://www.apache.org/licenses/LICENSE-2.0
9
+ #
10
+ # Unless required by applicable law or agreed to in writing, software
11
+ # distributed under the License is distributed on an "AS IS" BASIS,
12
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ # See the License for the specific language governing permissions and
14
+ # limitations under the License.
15
+
16
+ """
17
+ Hermes is a daemon that get the messages and sends them to external services (influxDB, ES, ActiveMQ).
18
+ """
19
+
20
+ import calendar
21
+ import datetime
22
+ import functools
23
+ import json
24
+ import logging
25
+ import random
26
+ import re
27
+ import smtplib
28
+ import socket
29
+ import sys
30
+ import threading
31
+ import time
32
+ from configparser import NoOptionError, NoSectionError
33
+ from email.mime.text import MIMEText
34
+ from typing import TYPE_CHECKING
35
+
36
+ import requests
37
+ import stomp
38
+
39
+ import rucio.db.sqla.util
40
+ from rucio.common.config import (
41
+ config_get,
42
+ config_get_bool,
43
+ config_get_int,
44
+ config_get_list,
45
+ )
46
+ from rucio.common.exception import DatabaseException
47
+ from rucio.common.logging import setup_logging
48
+ from rucio.core.message import delete_messages, retrieve_messages
49
+ from rucio.core.monitor import MetricManager
50
+ from rucio.daemons.common import run_daemon
51
+
52
+ if TYPE_CHECKING:
53
+ from collections.abc import Callable
54
+ from types import FrameType
55
+ from typing import Optional
56
+
57
+ from rucio.daemons.common import HeartbeatHandler
58
+
59
+ logging.getLogger("requests").setLevel(logging.CRITICAL)
60
+
61
+ METRICS = MetricManager(module=__name__)
62
+ graceful_stop = threading.Event()
63
+ DAEMON_NAME = "hermes"
64
+
65
+ RECONNECT_COUNTER = METRICS.counter(
66
+ name="reconnect.{host}",
67
+ documentation="Counts Hermes reconnects to different ActiveMQ brokers",
68
+ labelnames=("host",),
69
+ )
70
+
71
+
72
+ def default(datetype):
73
+ if isinstance(datetype, (datetime.date, datetime.datetime)):
74
+ return datetype.isoformat()
75
+
76
+
77
+ class HermesListener(stomp.ConnectionListener):
78
+ """
79
+ Hermes Listener
80
+ """
81
+
82
+ def __init__(self, broker):
83
+ """
84
+ __init__
85
+ """
86
+ self.__broker = broker
87
+
88
+ def on_error(self, frame):
89
+ """
90
+ Error handler
91
+ """
92
+ logging.error("[broker] [%s]: %s", self.__broker, frame.body)
93
+
94
+
95
+ def setup_activemq(logger: "Callable"):
96
+ """
97
+ Deliver messages to ActiveMQ
98
+
99
+ :param logger: The logger object.
100
+ """
101
+
102
+ logger(logging.INFO, "[broker] Resolving brokers")
103
+
104
+ brokers_alias = []
105
+ brokers_resolved = []
106
+ try:
107
+ brokers_alias = [
108
+ broker.strip()
109
+ for broker in config_get("messaging-hermes", "brokers").split(",")
110
+ ]
111
+ except:
112
+ raise Exception("Could not load brokers from configuration")
113
+
114
+ logger(logging.INFO, "[broker] Resolving broker dns alias: %s", brokers_alias)
115
+ brokers_resolved = []
116
+ for broker in brokers_alias:
117
+ try:
118
+ addrinfos = socket.getaddrinfo(
119
+ broker, 0, socket.AF_INET, 0, socket.IPPROTO_TCP
120
+ )
121
+ brokers_resolved.extend(ai[4][0] for ai in addrinfos)
122
+ except socket.gaierror as ex:
123
+ logger(
124
+ logging.ERROR,
125
+ "[broker] Cannot resolve domain name %s (%s)",
126
+ broker,
127
+ str(ex),
128
+ )
129
+
130
+ logger(logging.DEBUG, "[broker] Brokers resolved to %s", brokers_resolved)
131
+
132
+ if not brokers_resolved:
133
+ logger(logging.FATAL, "[broker] No brokers resolved.")
134
+ return None, None, None, None, None
135
+
136
+ broker_timeout = 3
137
+ if not broker_timeout: # Allow zero in config
138
+ broker_timeout = None
139
+
140
+ logger(logging.INFO, "[broker] Checking authentication method")
141
+ use_ssl = True
142
+ try:
143
+ use_ssl = config_get_bool("messaging-hermes", "use_ssl")
144
+ except:
145
+ logger(
146
+ logging.INFO,
147
+ "[broker] Could not find use_ssl in configuration -- please update your rucio.cfg",
148
+ )
149
+
150
+ port = config_get_int("messaging-hermes", "port")
151
+ vhost = config_get("messaging-hermes", "broker_virtual_host", raise_exception=False)
152
+ if not use_ssl:
153
+ username = config_get("messaging-hermes", "username")
154
+ password = config_get("messaging-hermes", "password")
155
+ port = config_get_int("messaging-hermes", "nonssl_port")
156
+
157
+ conns = []
158
+ for broker in brokers_resolved:
159
+ if not use_ssl:
160
+ logger(
161
+ logging.INFO,
162
+ "[broker] setting up username/password authentication: %s",
163
+ broker,
164
+ )
165
+ else:
166
+ logger(
167
+ logging.INFO,
168
+ "[broker] setting up ssl cert/key authentication: %s",
169
+ broker,
170
+ )
171
+
172
+ con = stomp.Connection12(
173
+ host_and_ports=[(broker, port)],
174
+ vhost=vhost,
175
+ keepalive=True,
176
+ timeout=broker_timeout,
177
+ )
178
+ if use_ssl:
179
+ con.set_ssl(
180
+ key_file=config_get("messaging-hermes", "ssl_key_file"),
181
+ cert_file=config_get("messaging-hermes", "ssl_cert_file"),
182
+ )
183
+
184
+ con.set_listener(
185
+ "rucio-hermes", HermesListener(con.transport._Transport__host_and_ports[0])
186
+ )
187
+
188
+ conns.append(con)
189
+ destination = config_get("messaging-hermes", "destination")
190
+ return conns, destination, username, password, use_ssl
191
+
192
+
193
+ def deliver_to_activemq(
194
+ messages, conns, destination, username, password, use_ssl, logger
195
+ ):
196
+ """
197
+ Deliver messages to ActiveMQ
198
+
199
+ :param messages: The list of messages.
200
+ :param conns: A list of connections.
201
+ :param destination: The destination topic or queue.
202
+ :param username: The username if no SSL connection.
203
+ :param password: The username if no SSL connection.
204
+ :param use_ssl: Boolean to choose if SSL connection is used.
205
+ :param logger: The logger object.
206
+
207
+ :returns: List of message_id to delete
208
+ """
209
+ to_delete = []
210
+ for message in messages:
211
+ try:
212
+ conn = random.sample(conns, 1)[0]
213
+ if not conn.is_connected():
214
+ host_and_ports = conn.transport._Transport__host_and_ports[0][0]
215
+ RECONNECT_COUNTER.labels(host=host_and_ports.split(".")[0]).inc()
216
+ if not use_ssl:
217
+ logger(
218
+ logging.INFO,
219
+ "[broker] - connecting with USERPASS to %s",
220
+ host_and_ports,
221
+ )
222
+ conn.connect(username, password, wait=True)
223
+ else:
224
+ logger(
225
+ logging.INFO,
226
+ "[broker] - connecting with SSL to %s",
227
+ host_and_ports,
228
+ )
229
+ conn.connect(wait=True)
230
+
231
+ conn.send(
232
+ body=json.dumps(
233
+ {
234
+ "event_type": str(message["event_type"]).lower(),
235
+ "payload": message["payload"],
236
+ "created_at": str(message["created_at"]),
237
+ }
238
+ ),
239
+ destination=destination,
240
+ headers={
241
+ "persistent": "true",
242
+ "event_type": str(message["event_type"]).lower(),
243
+ },
244
+ )
245
+
246
+ to_delete.append(message["id"])
247
+ except ValueError:
248
+ logger(
249
+ logging.ERROR,
250
+ "[broker] Cannot serialize payload to JSON: %s",
251
+ str(message["payload"]),
252
+ )
253
+ to_delete.append(message["id"])
254
+ continue
255
+ except stomp.exception.NotConnectedException as error:
256
+ logger(
257
+ logging.WARNING,
258
+ "[broker] Could not deliver message due to NotConnectedException: %s",
259
+ str(error),
260
+ )
261
+ continue
262
+ except stomp.exception.ConnectFailedException as error:
263
+ logger(
264
+ logging.WARNING,
265
+ "[broker] Could not deliver message due to ConnectFailedException: %s",
266
+ str(error),
267
+ )
268
+ continue
269
+ except Exception as error:
270
+ logger(logging.ERROR, "[broker] Could not deliver message: %s", str(error))
271
+ continue
272
+
273
+ if str(message["event_type"]).lower().startswith("transfer") or str(
274
+ message["event_type"]
275
+ ).lower().startswith("stagein"):
276
+ logger(
277
+ logging.DEBUG,
278
+ "[broker] - event_type: %s, scope: %s, name: %s, rse: %s, request-id: %s, transfer-id: %s, created_at: %s",
279
+ str(message["event_type"]).lower(),
280
+ message["payload"].get("scope", None),
281
+ message["payload"].get("name", None),
282
+ message["payload"].get("dst-rse", None),
283
+ message["payload"].get("request-id", None),
284
+ message["payload"].get("transfer-id", None),
285
+ str(message["created_at"]),
286
+ )
287
+
288
+ elif str(message["event_type"]).lower().startswith("dataset"):
289
+ logger(
290
+ logging.DEBUG,
291
+ "[broker] - event_type: %s, scope: %s, name: %s, rse: %s, rule-id: %s, created_at: %s)",
292
+ str(message["event_type"]).lower(),
293
+ message["payload"].get("scope", None),
294
+ message["payload"].get("name", None),
295
+ message["payload"].get("rse", None),
296
+ message["payload"].get("rule_id", None),
297
+ str(message["created_at"]),
298
+ )
299
+
300
+ elif str(message["event_type"]).lower().startswith("deletion"):
301
+ if "url" not in message["payload"]:
302
+ message["payload"]["url"] = "unknown"
303
+ logger(
304
+ logging.DEBUG,
305
+ "[broker] - event_type: %s, scope: %s, name: %s, rse: %s, url: %s, created_at: %s)",
306
+ str(message["event_type"]).lower(),
307
+ message["payload"].get("scope", None),
308
+ message["payload"].get("name", None),
309
+ message["payload"].get("rse", None),
310
+ message["payload"].get("url", None),
311
+ str(message["created_at"]),
312
+ )
313
+ else:
314
+ logger(logging.DEBUG, "[broker] Other message: %s", message)
315
+ return to_delete
316
+
317
+
318
+ def deliver_emails(messages: list[dict], logger: "Callable") -> list:
319
+ """
320
+ Sends emails
321
+
322
+ :param messages: The list of messages.
323
+ :param logger: The logger object.
324
+
325
+ :returns: List of message_id to delete
326
+ """
327
+
328
+ email_from = config_get("messaging-hermes", "email_from")
329
+ send_email = config_get_bool(
330
+ "messaging-hermes", "send_email", raise_exception=False, default=True
331
+ )
332
+ to_delete = []
333
+ for message in messages:
334
+ if message['event_type'] == 'email':
335
+ msg = MIMEText(message['payload']['body'])
336
+ msg['From'] = email_from
337
+ msg['To'] = ', '.join(message['payload']['to'])
338
+ msg['Subject'] = message['payload']['subject']
339
+
340
+ try:
341
+ if send_email:
342
+ smtp = smtplib.SMTP()
343
+ smtp.connect()
344
+ smtp.sendmail(
345
+ msg["From"], message["payload"]["to"], msg.as_string()
346
+ )
347
+ smtp.quit()
348
+ to_delete.append(message["id"])
349
+ except Exception as error:
350
+ logger(logging.ERROR, "Cannot send email : %s", str(error))
351
+ else:
352
+ to_delete.append(message["id"])
353
+ continue
354
+ return to_delete
355
+
356
+
357
+ def submit_to_elastic(messages: list[dict], endpoint: str, logger: "Callable") -> int:
358
+ """
359
+ Aggregate a list of message to ElasticSearch
360
+
361
+ :param messages: The list of messages.
362
+ :param endpoint: The ES endpoint were to send the messages.
363
+ :param logger: The logger object.
364
+
365
+ :returns: HTTP status code. 200 and 204 OK. Rest is failure.
366
+ """
367
+ text = ""
368
+ for message in messages:
369
+ text += '{ "index":{ } }\n%s\n' % json.dumps(message, default=default)
370
+ res = requests.post(
371
+ endpoint, data=text, headers={"Content-Type": "application/json"}
372
+ )
373
+ return res.status_code
374
+
375
+
376
+ def aggregate_to_influx(
377
+ messages: list[dict], bin_size: int, endpoint: str, logger: "Callable"
378
+ ) -> int:
379
+ """
380
+ Aggregate a list of message using a certain bin_size
381
+ and submit them to a InfluxDB endpoint
382
+
383
+ :param messages: The list of messages.
384
+ :param bin_size: The size of the bins for the aggreagation (e.g. 10m, 1h, etc.).
385
+ :param endpoint: The InfluxDB endpoint were to send the messages.
386
+ :param logger: The logger object.
387
+
388
+ :returns: HTTP status code. 200 and 204 OK. Rest is failure.
389
+ """
390
+ bins = {}
391
+ dtime = datetime.datetime.now()
392
+ microsecond = dtime.microsecond
393
+
394
+ for message in messages:
395
+ event_type = message["event_type"]
396
+ payload = message["payload"]
397
+ if event_type in ["transfer-failed", "transfer-done"]:
398
+ if not payload["transferred_at"]:
399
+ logger(
400
+ logging.WARNING,
401
+ "No transferred_at for message. Reason : %s",
402
+ payload["reason"],
403
+ )
404
+ continue
405
+ transferred_at = time.strptime(
406
+ payload["transferred_at"], "%Y-%m-%d %H:%M:%S"
407
+ )
408
+ if bin_size == "1m":
409
+ transferred_at = int(calendar.timegm(transferred_at)) * 1000000000
410
+ transferred_at += microsecond
411
+ if transferred_at not in bins:
412
+ bins[transferred_at] = {}
413
+ src_rse, dest_rse, activity = (
414
+ payload["src-rse"],
415
+ payload["dst-rse"],
416
+ payload["activity"],
417
+ )
418
+ activity = re.sub(" ", r"\ ", activity)
419
+ key = "transfer,activity=%s,src_rse=%s,dst_rse=%s" % (
420
+ activity,
421
+ src_rse,
422
+ dest_rse,
423
+ )
424
+ if key not in bins[transferred_at]:
425
+ bins[transferred_at][key] = [0, 0, 0, 0]
426
+ if event_type == "transfer-done":
427
+ bins[transferred_at][key][0] += 1
428
+ bins[transferred_at][key][1] += payload["bytes"]
429
+ if event_type == "transfer-failed":
430
+ bins[transferred_at][key][2] += 1
431
+ bins[transferred_at][key][3] += payload["bytes"]
432
+ elif event_type in ["deletion-failed", "deletion-done"]:
433
+ created_at = message["created_at"]
434
+ if bin_size == "1m":
435
+ created_at = created_at.replace(
436
+ second=0, microsecond=0, tzinfo=datetime.timezone.utc
437
+ ).timestamp()
438
+ created_at = int(created_at) * 1000000000
439
+ created_at += microsecond
440
+ if created_at not in bins:
441
+ bins[created_at] = {}
442
+ rse = payload["rse"]
443
+ key = "deletion,rse=%s" % (rse)
444
+ if key not in bins[created_at]:
445
+ bins[created_at][key] = [0, 0, 0, 0]
446
+ if event_type == "deletion-done":
447
+ bins[created_at][key][0] += 1
448
+ bins[created_at][key][1] += payload["bytes"]
449
+ if event_type == "deletion-failed":
450
+ bins[created_at][key][2] += 1
451
+ bins[created_at][key][3] += payload["bytes"]
452
+ points = ""
453
+ for timestamp in bins:
454
+ for entry in bins[timestamp]:
455
+ metrics = bins[timestamp][entry]
456
+ event_type = entry.split(",")[0]
457
+ point = (
458
+ "%s nb_%s_done=%s,bytes_%s_done=%s,nb_%s_failed=%s,bytes_%s_failed=%s %s"
459
+ % (
460
+ entry,
461
+ event_type,
462
+ metrics[0],
463
+ event_type,
464
+ metrics[1],
465
+ event_type,
466
+ metrics[2],
467
+ event_type,
468
+ metrics[3],
469
+ timestamp,
470
+ )
471
+ )
472
+ points += point
473
+ points += "\n"
474
+ influx_token = config_get("hermes", "influxdb_token", False, None)
475
+ if influx_token:
476
+ headers = {"Authorization": "Token %s" % influx_token}
477
+ if points:
478
+ res = requests.post(endpoint, headers=headers, data=points)
479
+ logger(logging.DEBUG, "%s", str(res.text))
480
+ return res.status_code
481
+ return 204
482
+
483
+
484
+ def hermes(once: bool = False, bulk: int = 1000, sleep_time: int = 10) -> None:
485
+ """
486
+ Creates a Hermes Worker that can submit messages to different services (InfluXDB, ElasticSearch, ActiveMQ)
487
+ The list of services need to be define in the config service in the hermes section.
488
+ The list of endpoints need to be defined in rucio.cfg in the hermes section.
489
+
490
+ :param once: Run only once.
491
+ :param bulk: The number of requests to process.
492
+ :param sleep_time: Time between two cycles.
493
+ """
494
+ run_daemon(
495
+ once=once,
496
+ graceful_stop=graceful_stop,
497
+ executable=DAEMON_NAME,
498
+ partition_wait_time=1,
499
+ sleep_time=sleep_time,
500
+ run_once_fnc=functools.partial(
501
+ run_once,
502
+ bulk=bulk,
503
+ ),
504
+ )
505
+
506
+
507
+ def run_once(heartbeat_handler: "HeartbeatHandler", bulk: int, **_kwargs) -> bool:
508
+
509
+ worker_number, total_workers, logger = heartbeat_handler.live()
510
+ try:
511
+ services_list = config_get_list("hermes", "services_list")
512
+ except (NoOptionError, NoSectionError, RuntimeError):
513
+ logger(logging.DEBUG, "No services found, exiting")
514
+ sys.exit(1)
515
+
516
+ if "influx" in services_list:
517
+ influx_endpoint = None
518
+ try:
519
+ influx_endpoint = config_get("hermes", "influxdb_endpoint", False, None)
520
+ if not influx_endpoint:
521
+ logger(
522
+ logging.ERROR,
523
+ "InfluxDB defined in the services list, but no endpoint can be found",
524
+ )
525
+ except Exception as err:
526
+ logger(logging.ERROR, str(err))
527
+ if "elastic" in services_list:
528
+ elastic_endpoint = None
529
+ try:
530
+ elastic_endpoint = config_get("hermes", "elastic_endpoint", False, None)
531
+ if not elastic_endpoint:
532
+ logger(
533
+ logging.ERROR,
534
+ "Elastic defined in the services list, but no endpoint can be found",
535
+ )
536
+ except Exception as err:
537
+ logger(logging.ERROR, str(err))
538
+ conns = None
539
+ if "activemq" in services_list:
540
+ try:
541
+ conns, destination, username, password, use_ssl = setup_activemq(logger)
542
+ if not conns:
543
+ logger(
544
+ logging.ERROR,
545
+ "ActiveMQ defined in the services list, cannot be setup",
546
+ )
547
+ except Exception as err:
548
+ logger(logging.ERROR, str(err))
549
+
550
+ worker_number, total_workers, logger = heartbeat_handler.live()
551
+ message_dict = {}
552
+ message_ids = []
553
+ start_time = time.time()
554
+ messages = retrieve_messages(
555
+ bulk=bulk,
556
+ old_mode=False,
557
+ thread=worker_number,
558
+ total_threads=total_workers,
559
+ )
560
+
561
+ to_delete = []
562
+ if messages:
563
+ for message in messages:
564
+ service = message["services"]
565
+ if service not in message_dict:
566
+ message_dict[service] = []
567
+ message_dict[service].append(message)
568
+ message_ids.append(message["id"])
569
+ logger(
570
+ logging.DEBUG,
571
+ "Retrieved %i messages retrieved in %s seconds",
572
+ len(messages),
573
+ time.time() - start_time,
574
+ )
575
+
576
+ if "influx" in message_dict and influx_endpoint:
577
+ # For influxDB, bulk submission, either everything succeeds or fails
578
+ t_time = time.time()
579
+ logger(logging.DEBUG, "Will submit to influxDB")
580
+ try:
581
+ state = aggregate_to_influx(
582
+ messages=message_dict["influx"],
583
+ bin_size="1m",
584
+ endpoint=influx_endpoint,
585
+ logger=logger,
586
+ )
587
+ if state in [204, 200]:
588
+ logger(
589
+ logging.INFO,
590
+ "%s messages successfully submitted to influxDB in %s seconds",
591
+ len(message_dict["influx"]),
592
+ time.time() - t_time,
593
+ )
594
+ for message in message_dict["influx"]:
595
+ to_delete.append(message)
596
+ else:
597
+ logger(
598
+ logging.ERROR,
599
+ "Failure to submit %s messages to influxDB. Returned status: %s",
600
+ len(message_dict["influx"]),
601
+ state,
602
+ )
603
+ except Exception as error:
604
+ logger(logging.ERROR, "Error sending to InfluxDB : %s", str(error))
605
+
606
+ if "elastic" in message_dict and elastic_endpoint:
607
+ # For elastic, bulk submission, either everything succeeds or fails
608
+ t_time = time.time()
609
+ try:
610
+ state = submit_to_elastic(
611
+ messages=message_dict["elastic"],
612
+ endpoint=elastic_endpoint,
613
+ logger=logger,
614
+ )
615
+ if state in [200, 204]:
616
+ logger(
617
+ logging.INFO,
618
+ "%s messages successfully submitted to elastic in %s seconds",
619
+ len(message_dict["elastic"]),
620
+ time.time() - t_time,
621
+ )
622
+ for message in message_dict["elastic"]:
623
+ to_delete.append(message)
624
+ else:
625
+ logger(
626
+ logging.ERROR,
627
+ "Failure to submit %s messages to elastic. Returned status: %s",
628
+ len(message_dict["influx"]),
629
+ state,
630
+ )
631
+ except Exception as error:
632
+ logger(logging.ERROR, "Error sending to Elastic : %s", str(error))
633
+
634
+ if "email" in message_dict:
635
+ t_time = time.time()
636
+ try:
637
+ messages_sent = deliver_emails(
638
+ messages=message_dict["email"], logger=logger
639
+ )
640
+ logger(
641
+ logging.INFO,
642
+ "%s messages successfully submitted by emails in %s seconds",
643
+ len(message_dict["email"]),
644
+ time.time() - t_time,
645
+ )
646
+ for message in message_dict["email"]:
647
+ if message["id"] in messages_sent:
648
+ to_delete.append(message)
649
+ except Exception as error:
650
+ logger(logging.ERROR, "Error sending email : %s", str(error))
651
+
652
+ if "activemq" in message_dict and conns:
653
+ t_time = time.time()
654
+ try:
655
+ messages_sent = deliver_to_activemq(
656
+ messages=message_dict["activemq"],
657
+ conns=conns,
658
+ destination=destination,
659
+ username=username,
660
+ password=password,
661
+ use_ssl=use_ssl,
662
+ logger=logger,
663
+ )
664
+ logger(
665
+ logging.INFO,
666
+ "%s messages successfully submitted to ActiveMQ in %s seconds",
667
+ len(message_dict["activemq"]),
668
+ time.time() - t_time,
669
+ )
670
+ for message in message_dict["activemq"]:
671
+ if message["id"] in messages_sent:
672
+ to_delete.append(message)
673
+ except Exception as error:
674
+ logger(logging.ERROR, "Error sending to ActiveMQ : %s", str(error))
675
+
676
+ logger(logging.INFO, "Deleting %s messages", len(to_delete))
677
+ to_delete = [
678
+ {
679
+ "id": message["id"],
680
+ "created_at": message["created_at"],
681
+ "updated_at": message["created_at"],
682
+ "payload": str(message["payload"]),
683
+ "event_type": message["event_type"],
684
+ }
685
+ for message in to_delete
686
+ ]
687
+ delete_messages(messages=to_delete)
688
+ must_sleep = True
689
+ return must_sleep
690
+
691
+
692
+ def stop(signum: "Optional[int]" = None, frame: "Optional[FrameType]" = None) -> None:
693
+ """
694
+ Graceful exit.
695
+ """
696
+ logging.info("Caught CTRL-C - waiting for cycle to end before shutting down")
697
+ graceful_stop.set()
698
+
699
+
700
+ def run(
701
+ once: bool = False,
702
+ threads: int = 1,
703
+ bulk: int = 1000,
704
+ sleep_time: int = 10,
705
+ broker_timeout: int = 3,
706
+ ) -> None:
707
+ """
708
+ Starts up the hermes threads.
709
+ """
710
+ setup_logging(process_name=DAEMON_NAME)
711
+
712
+ if rucio.db.sqla.util.is_old_db():
713
+ raise DatabaseException("Database was not updated, daemon won't start")
714
+
715
+ logging.info("starting hermes threads")
716
+ thread_list = [
717
+ threading.Thread(
718
+ target=hermes,
719
+ kwargs={
720
+ "once": once,
721
+ "bulk": bulk,
722
+ "sleep_time": sleep_time,
723
+ },
724
+ )
725
+ for _ in range(0, threads)
726
+ ]
727
+
728
+ for thrd in thread_list:
729
+ thrd.start()
730
+
731
+ logging.debug(thread_list)
732
+ # Interruptible joins require a timeout.
733
+ while thread_list:
734
+ thread_list = [
735
+ thread.join(timeout=3.14)
736
+ for thread in thread_list
737
+ if thread and thread.is_alive()
738
+ ]