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,14 @@
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.
@@ -0,0 +1,513 @@
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
+ This daemon consumes tracer messages from ActiveMQ and updates the atime for replicas.
18
+ """
19
+
20
+ import functools
21
+ import logging
22
+ import re
23
+ from configparser import NoOptionError, NoSectionError
24
+ from datetime import datetime
25
+ from json import loads as jloads, dumps as jdumps
26
+ from queue import Queue
27
+ from threading import Event, Thread
28
+ from time import time
29
+ from typing import TYPE_CHECKING
30
+
31
+ import rucio.db.sqla.util
32
+ from rucio.common.config import config_get, config_get_bool, config_get_int, config_get_list
33
+ from rucio.common.exception import RSENotFound, DatabaseException
34
+ from rucio.common.logging import setup_logging
35
+ from rucio.common.stomp_utils import StompConnectionManager
36
+ from rucio.common.stopwatch import Stopwatch
37
+ from rucio.common.types import InternalAccount, InternalScope
38
+ from rucio.core.did import touch_dids, list_parent_dids
39
+ from rucio.core.lock import touch_dataset_locks
40
+ from rucio.core.monitor import MetricManager
41
+ from rucio.core.replica import touch_replica, touch_collection_replicas, declare_bad_file_replicas
42
+ from rucio.core.rse import get_rse_id
43
+ from rucio.daemons.common import HeartbeatHandler, run_daemon
44
+ from rucio.db.sqla.constants import DIDType, BadFilesStatus
45
+
46
+ if TYPE_CHECKING:
47
+ from types import FrameType
48
+ from typing import Optional
49
+
50
+
51
+ logging.getLogger("stomp").setLevel(logging.CRITICAL)
52
+
53
+ METRICS = MetricManager(module=__name__)
54
+ graceful_stop = Event()
55
+
56
+
57
+ class AMQConsumer:
58
+ """ActiveMQ message consumer"""
59
+
60
+ def __init__(self, broker, conn, queue, chunksize, subscription_id, excluded_usrdns, dataset_queue, bad_files_patterns, logger=logging.log):
61
+ self.__broker = broker
62
+ self.__conn = conn
63
+ self.__queue = queue
64
+ self.__reports = []
65
+ self.__ids = []
66
+ self.__chunksize = chunksize
67
+ self.__subscription_id = subscription_id
68
+ # excluded states empty for the moment, maybe that should be recosidered in the future
69
+ self.__excluded_states = set([])
70
+ # exclude specific usrdns like GangaRBT
71
+ self.__excluded_usrdns = excluded_usrdns
72
+ self.__dataset_queue = dataset_queue
73
+ self.__bad_files_patterns = bad_files_patterns
74
+ self.__logger = logger
75
+
76
+ @METRICS.count_it
77
+ def on_heartbeat_timeout(self):
78
+ self.__conn.disconnect()
79
+
80
+ @METRICS.count_it
81
+ def on_error(self, frame):
82
+ self.__logger(logging.ERROR, 'Message receive error: [%s] %s' % (self.__broker, frame.body))
83
+
84
+ @METRICS.count_it
85
+ def on_message(self, frame):
86
+ appversion = 'dq2'
87
+ msg_id = frame.headers['message-id']
88
+ if 'appversion' in frame.headers:
89
+ appversion = frame.headers['appversion']
90
+
91
+ if 'resubmitted' in frame.headers:
92
+ METRICS.counter('received_resubmitted').inc()
93
+
94
+ try:
95
+ if appversion == 'dq2':
96
+ self.__conn.ack(msg_id, self.__subscription_id)
97
+ return
98
+ else:
99
+ report = jloads(frame.body)
100
+ except Exception:
101
+ # message is corrupt, not much to do here
102
+ # send count to graphite, send ack to broker and return
103
+ METRICS.counter('json_error').inc()
104
+ self.__logger(logging.ERROR, 'json error', exc_info=True)
105
+ self.__conn.ack(msg_id, self.__subscription_id)
106
+ return
107
+
108
+ self.__ids.append(msg_id)
109
+ self.__reports.append(report)
110
+
111
+ try:
112
+ self.__logger(logging.DEBUG, 'message received: %s %s %s' % (str(report['eventType']), report['filename'], report['remoteSite']))
113
+ except Exception:
114
+ pass
115
+
116
+ if len(self.__ids) >= self.__chunksize:
117
+ self.__update_atime()
118
+ for msg_id in self.__ids:
119
+ self.__conn.ack(msg_id, self.__subscription_id)
120
+
121
+ self.__reports = []
122
+ self.__ids = []
123
+
124
+ def __update_atime(self):
125
+ """
126
+ Bulk update atime.
127
+ """
128
+ replicas = []
129
+ rses = []
130
+ for report in self.__reports:
131
+ if 'vo' not in report:
132
+ report['vo'] = 'def'
133
+
134
+ try:
135
+ # Identify suspicious files
136
+ try:
137
+ if self.__bad_files_patterns and report['eventType'] in ['get_sm', 'get_sm_a', 'get'] and 'clientState' in report and report['clientState'] not in ['DONE', 'FOUND_ROOT', 'ALREADY_DONE']:
138
+ for pattern in self.__bad_files_patterns:
139
+ if 'stateReason' in report and report['stateReason'] and isinstance(report['stateReason'], str) and pattern.match(report['stateReason']):
140
+ reason = report['stateReason'][:255]
141
+ if 'url' not in report or not report['url']:
142
+ self.__logger(logging.ERROR, 'Missing url in the following trace : ' + str(report))
143
+ else:
144
+ try:
145
+ surl = report['url']
146
+ declare_bad_file_replicas([surl, ], reason=reason, issuer=InternalAccount('root', vo=report['vo']), status=BadFilesStatus.SUSPICIOUS)
147
+ self.__logger(logging.INFO, 'Declare suspicious file %s with reason %s' % (report['url'], reason))
148
+ except Exception as error:
149
+ self.__logger(logging.ERROR, 'Failed to declare suspicious file' + str(error))
150
+ except Exception as error:
151
+ self.__logger(logging.ERROR, 'Problem with bad trace : %s . Error %s' % (str(report), str(error)))
152
+
153
+ # check if scope in report. if not skip this one.
154
+ if 'scope' not in report:
155
+ METRICS.counter('missing_scope').inc()
156
+ if report['eventType'] != 'touch':
157
+ continue
158
+ else:
159
+ METRICS.counter('with_scope').inc()
160
+ report['scope'] = InternalScope(report['scope'], report['vo'])
161
+
162
+ # handle all events starting with get* and download and touch events.
163
+ if not report['eventType'].startswith('get') and not report['eventType'].startswith('sm_get') and not report['eventType'] == 'download' and not report['eventType'] == 'touch':
164
+ continue
165
+ if report['eventType'].endswith('_es'):
166
+ continue
167
+ METRICS.counter('total_get').inc()
168
+ if report['eventType'] == 'get':
169
+ METRICS.counter('dq2clients').inc()
170
+ elif report['eventType'] == 'get_sm' or report['eventType'] == 'sm_get':
171
+ if report['eventVersion'] == 'aCT':
172
+ METRICS.counter('panda_production_act').inc()
173
+ else:
174
+ METRICS.counter('panda_production').inc()
175
+ elif report['eventType'] == 'get_sm_a' or report['eventType'] == 'sm_get_a':
176
+ if report['eventVersion'] == 'aCT':
177
+ METRICS.counter('panda_analysis_act').inc()
178
+ else:
179
+ METRICS.counter('panda_analysis').inc()
180
+ elif report['eventType'] == 'download':
181
+ METRICS.counter('rucio_download').inc()
182
+ elif report['eventType'] == 'touch':
183
+ METRICS.counter('rucio_touch').inc()
184
+ else:
185
+ METRICS.counter('other_get').inc()
186
+
187
+ if report['eventType'] == 'download' or report['eventType'] == 'touch':
188
+ report['usrdn'] = report['account']
189
+
190
+ if report['usrdn'] in self.__excluded_usrdns:
191
+ continue
192
+ # handle touch and non-touch traces differently
193
+ if report['eventType'] != 'touch':
194
+ # check if the report has the right state.
195
+ if 'eventVersion' in report:
196
+ if report['eventVersion'] != 'aCT':
197
+ if report['clientState'] in self.__excluded_states:
198
+ continue
199
+
200
+ if 'remoteSite' not in report:
201
+ continue
202
+ if not report['remoteSite']:
203
+ continue
204
+
205
+ if 'filename' not in report:
206
+ if 'name' in report:
207
+ report['filename'] = report['name']
208
+
209
+ rses = report['remoteSite'].strip().split(',')
210
+ for rse in rses:
211
+ try:
212
+ rse_id = get_rse_id(rse=rse, vo=report['vo'])
213
+ except RSENotFound:
214
+ self.__logger(logging.WARNING, "Cannot lookup rse_id for %s. Will skip this report.", rse)
215
+ METRICS.counter('rse_not_found').inc()
216
+ continue
217
+ replicas.append({'name': report['filename'], 'scope': report['scope'], 'rse': rse, 'rse_id': rse_id, 'accessed_at': datetime.utcfromtimestamp(report['traceTimeentryUnix']),
218
+ 'traceTimeentryUnix': report['traceTimeentryUnix'], 'eventVersion': report['eventVersion']})
219
+ else:
220
+ # if touch event and if datasetScope is in the report then it means
221
+ # that there is no file scope/name and therefore only the dataset is
222
+ # put in the queue to be updated and the rest is skipped.
223
+ rse_id = None
224
+ rse = None
225
+ if 'remoteSite' in report:
226
+ rse = report['remoteSite']
227
+ try:
228
+ rse_id = get_rse_id(rse=rse, vo=report['vo'])
229
+ except RSENotFound:
230
+ self.__logger(logging.WARNING, "Cannot lookup rse_id for %s.", rse)
231
+ METRICS.counter('rse_not_found').inc()
232
+ if 'datasetScope' in report:
233
+ self.__dataset_queue.put({'scope': InternalScope(report['datasetScope'], vo=report['vo']),
234
+ 'name': report['dataset'],
235
+ 'rse_id': rse_id,
236
+ 'accessed_at': datetime.utcfromtimestamp(report['traceTimeentryUnix'])})
237
+ continue
238
+ else:
239
+ if 'remoteSite' not in report:
240
+ continue
241
+ replicas.append({'name': report['filename'],
242
+ 'scope': report['scope'],
243
+ 'rse': rse,
244
+ 'rse_id': rse_id,
245
+ 'accessed_at': datetime.utcfromtimestamp(report['traceTimeentryUnix'])})
246
+
247
+ except (KeyError, AttributeError):
248
+ self.__logger(logging.ERROR, "Cannot handle report.", exc_info=True)
249
+ METRICS.counter('report_error').inc()
250
+ continue
251
+ except Exception:
252
+ self.__logger(logging.ERROR, "Exception", exc_info=True)
253
+ continue
254
+
255
+ for did in list_parent_dids(report['scope'], report['filename']):
256
+ if did['type'] != DIDType.DATASET:
257
+ continue
258
+ # do not update _dis datasets
259
+ if did['scope'].external == 'panda' and '_dis' in did['name']:
260
+ continue
261
+ for rse in rses:
262
+ try:
263
+ rse_id = get_rse_id(rse=rse, vo=report['vo'])
264
+ except RSENotFound:
265
+ self.__logger(logging.WARNING, "Cannot lookup rse_id for %s. Will skip this report.", rse)
266
+ METRICS.counter('rse_not_found').inc()
267
+ continue
268
+ self.__dataset_queue.put({'scope': did['scope'], 'name': did['name'], 'did_type': did['type'], 'rse_id': rse_id, 'accessed_at': datetime.utcfromtimestamp(report['traceTimeentryUnix'])})
269
+
270
+ if not len(replicas):
271
+ return
272
+
273
+ self.__logger(logging.DEBUG, "trying to update replicas: %s", replicas)
274
+
275
+ stopwatch = Stopwatch()
276
+ try:
277
+ for replica in replicas:
278
+ # if touch replica hits a locked row put the trace back into queue for later retry
279
+ if not touch_replica(replica):
280
+ resubmit = {'filename': replica['name'],
281
+ 'scope': replica['scope'].external,
282
+ 'remoteSite': replica['rse'],
283
+ 'traceTimeentryUnix': replica['traceTimeentryUnix'],
284
+ 'eventType': 'get',
285
+ 'usrdn': 'someuser',
286
+ 'clientState': 'DONE',
287
+ 'eventVersion': replica['eventVersion']}
288
+ if replica['scope'].vo != 'def':
289
+ resubmit['vo'] = replica['scope'].vo
290
+ self.__conn.send(body=jdumps(resubmit), destination=self.__queue, headers={'appversion': 'rucio', 'resubmitted': '1'})
291
+ METRICS.counter('sent_resubmitted').inc()
292
+ METRICS.timer('update_atime').observe(stopwatch.elapsed)
293
+ except Exception:
294
+ self.__logger(logging.ERROR, "Cannot update replicas.", exc_info=True)
295
+ METRICS.counter('update_error').inc()
296
+
297
+ METRICS.counter('updated_replicas').inc()
298
+
299
+
300
+ def kronos_file(once: bool = False, dataset_queue: Queue = None, sleep_time: int = 60):
301
+ """
302
+ Main loop to consume tracer reports.
303
+ """
304
+ stomp_conn_mngr = StompConnectionManager()
305
+ run_daemon(
306
+ once=once,
307
+ graceful_stop=graceful_stop,
308
+ executable='kronos-file',
309
+ partition_wait_time=1,
310
+ sleep_time=sleep_time,
311
+ run_once_fnc=functools.partial(
312
+ run_once_kronos_file,
313
+ stomp_conn_mngr=stomp_conn_mngr,
314
+ dataset_queue=dataset_queue,
315
+ sleep_time=sleep_time,
316
+ )
317
+ )
318
+ stomp_conn_mngr.disconnect()
319
+
320
+
321
+ def run_once_kronos_file(heartbeat_handler: HeartbeatHandler, stomp_conn_mngr: StompConnectionManager, dataset_queue: Queue, sleep_time: int, **kwargs):
322
+ """
323
+ Run the amq consumer once.
324
+ """
325
+ _, _, logger = heartbeat_handler.live()
326
+
327
+ chunksize = config_get_int('tracer-kronos', 'chunksize')
328
+ prefetch_size = config_get_int('tracer-kronos', 'prefetch_size')
329
+ subscription_id = config_get('tracer-kronos', 'subscription_id')
330
+ # Load bad file patterns from config
331
+ try:
332
+ bad_files_patterns = []
333
+ pattern = config_get(section='kronos', option='bad_files_patterns', session=None)
334
+ pattern = str(pattern)
335
+ patterns = pattern.split(",")
336
+ for pat in patterns:
337
+ bad_files_patterns.append(re.compile(pat.strip()))
338
+ except (NoOptionError, NoSectionError, RuntimeError):
339
+ bad_files_patterns = []
340
+ except Exception as error:
341
+ logger.error(f'Failed to get bad_file_patterns {str(error)}')
342
+ bad_files_patterns = []
343
+
344
+ use_ssl = config_get_bool('tracer-kronos', 'use_ssl', default=True, raise_exception=False)
345
+ if not use_ssl:
346
+ username = config_get('tracer-kronos', 'username')
347
+ password = config_get('tracer-kronos', 'password')
348
+
349
+ excluded_usrdns = set(config_get('tracer-kronos', 'excluded_usrdns').split(','))
350
+ vhost = config_get('tracer-kronos', 'broker_virtual_host', raise_exception=False)
351
+
352
+ brokers_alias = config_get_list('tracer-kronos', 'brokers')
353
+ port = config_get_int('tracer-kronos', 'port')
354
+ reconnect_attempts = config_get_int('tracer-kronos', 'reconnect_attempts')
355
+ ssl_key_file = config_get('tracer-kronos', 'ssl_key_file', raise_exception=False)
356
+ ssl_cert_file = config_get('tracer-kronos', 'ssl_cert_file', raise_exception=False)
357
+
358
+ created_conns, _ = stomp_conn_mngr.re_configure(
359
+ brokers=brokers_alias,
360
+ port=port,
361
+ use_ssl=use_ssl,
362
+ vhost=vhost,
363
+ reconnect_attempts=reconnect_attempts,
364
+ ssl_key_file=ssl_key_file,
365
+ ssl_cert_file=ssl_cert_file,
366
+ timeout=sleep_time,
367
+ heartbeats=(0, 5000),
368
+ logger=logger,
369
+ )
370
+
371
+ for conn in created_conns:
372
+ if not conn.is_connected():
373
+ logger(logging.INFO, 'connecting to %s' % str(conn.transport._Transport__host_and_ports[0]))
374
+ METRICS.counter('reconnect.{host}').labels(host=conn.transport._Transport__host_and_ports[0][0]).inc()
375
+ conn.set_listener('rucio-tracer-kronos', AMQConsumer(broker=conn.transport._Transport__host_and_ports[0],
376
+ conn=conn,
377
+ queue=config_get('tracer-kronos', 'queue'),
378
+ chunksize=chunksize,
379
+ subscription_id=subscription_id,
380
+ excluded_usrdns=excluded_usrdns,
381
+ dataset_queue=dataset_queue,
382
+ bad_files_patterns=bad_files_patterns,
383
+ logger=logger))
384
+ if not use_ssl:
385
+ conn.connect(username, password)
386
+ else:
387
+ conn.connect()
388
+ conn.subscribe(destination=config_get('tracer-kronos', 'queue'), ack='client-individual', id=subscription_id, headers={'activemq.prefetchSize': prefetch_size})
389
+
390
+
391
+ def kronos_dataset(dataset_queue: Queue, once: bool = False, sleep_time: int = 60):
392
+ return_values = {'heartbeat_handler': HeartbeatHandler("kronos-dataset", 10)}
393
+ run_daemon(
394
+ once=once,
395
+ graceful_stop=graceful_stop,
396
+ executable='kronos-dataset',
397
+ partition_wait_time=1,
398
+ sleep_time=sleep_time,
399
+ run_once_fnc=functools.partial(
400
+ run_once_kronos_dataset,
401
+ dataset_queue=dataset_queue,
402
+ return_values=return_values,
403
+ )
404
+ )
405
+
406
+ # once again for potential backlog
407
+ run_once_kronos_dataset(dataset_queue=dataset_queue, return_values=return_values, heartbeat_handler=return_values['heartbeat_handler'], sleep_time=sleep_time)
408
+
409
+
410
+ def run_once_kronos_dataset(dataset_queue: Queue, return_values: dict, heartbeat_handler: HeartbeatHandler, **kwargs):
411
+ if heartbeat_handler is None:
412
+ if "heartbeat_handler" not in return_values.keys():
413
+ return_values["heartbeat_handler"] = HeartbeatHandler("kronos-dataset", 10)
414
+ heartbeat_handler = return_values["heartbeat_handler"]
415
+
416
+ _, _, logger = heartbeat_handler.live()
417
+ len_ds = dataset_queue.qsize()
418
+ datasets = {}
419
+ dslocks = {}
420
+ now = time()
421
+ for _ in range(0, len_ds):
422
+ dataset = dataset_queue.get()
423
+ did = '%s:%s' % (dataset['scope'].internal, dataset['name'])
424
+ rse = dataset['rse_id']
425
+ if did not in datasets:
426
+ datasets[did] = dataset['accessed_at']
427
+ else:
428
+ datasets[did] = max(datasets[did], dataset['accessed_at'])
429
+
430
+ if rse is None:
431
+ continue
432
+ if did not in dslocks:
433
+ dslocks[did] = {}
434
+ if rse not in dslocks[did]:
435
+ dslocks[did][rse] = dataset['accessed_at']
436
+ else:
437
+ dslocks[did][rse] = max(dataset['accessed_at'], dslocks[did][rse])
438
+ logger(logging.INFO, 'fetched %d datasets from queue (%ds)' % (len_ds, time() - now))
439
+
440
+ total, failed, start = 0, 0, time()
441
+ for did, accessed_at in datasets.items():
442
+ scope, name = did.split(':')
443
+ scope = InternalScope(scope, fromExternal=False)
444
+ update_did = {'scope': scope, 'name': name, 'type': DIDType.DATASET, 'accessed_at': accessed_at}
445
+ # if update fails, put back in queue and retry next time
446
+ if not touch_dids((update_did,)):
447
+ update_did['rse_id'] = None
448
+ dataset_queue.put(update_did)
449
+ failed += 1
450
+ total += 1
451
+ logger(logging.INFO, 'update done for %d datasets, %d failed (%ds)' % (total, failed, time() - start))
452
+
453
+ total, failed, start = 0, 0, time()
454
+ for did, rses in dslocks.items():
455
+ scope, name = did.split(':')
456
+ scope = InternalScope(scope, fromExternal=False)
457
+ for rse, accessed_at in rses.items():
458
+ update_dslock = {'scope': scope, 'name': name, 'rse_id': rse, 'accessed_at': accessed_at}
459
+ # if update fails, put back in queue and retry next time
460
+ if not touch_dataset_locks((update_dslock,)):
461
+ dataset_queue.put(update_dslock)
462
+ failed += 1
463
+ total += 1
464
+ logger(logging.INFO, 'update done for %d locks, %d failed (%ds)' % (total, failed, time() - start))
465
+
466
+ total, failed, start = 0, 0, time()
467
+ for did, rses in dslocks.items():
468
+ scope, name = did.split(':')
469
+ scope = InternalScope(scope, fromExternal=False)
470
+ for rse, accessed_at in rses.items():
471
+ update_dslock = {'scope': scope, 'name': name, 'rse_id': rse, 'accessed_at': accessed_at}
472
+ # if update fails, put back in queue and retry next time
473
+ if not touch_collection_replicas((update_dslock,)):
474
+ dataset_queue.put(update_dslock)
475
+ failed += 1
476
+ total += 1
477
+ logger(logging.INFO, 'update done for %d collection replicas, %d failed (%ds)' % (total, failed, time() - start))
478
+
479
+
480
+ def stop(signum: "Optional[int]" = None, frame: "Optional[FrameType]" = None) -> None:
481
+ """
482
+ Graceful exit.
483
+ """
484
+ graceful_stop.set()
485
+
486
+
487
+ def run(once=False, threads=1, sleep_time_datasets=60, sleep_time_files=60):
488
+ """
489
+ Starts up the consumer threads
490
+ """
491
+ setup_logging(process_name='tracer-kronos')
492
+
493
+ if rucio.db.sqla.util.is_old_db():
494
+ raise DatabaseException('Database was not updated, daemon won\'t start')
495
+
496
+ dataset_queue = Queue()
497
+ logging.info('starting tracer consumer threads')
498
+
499
+ thread_list = []
500
+ for _ in range(0, threads):
501
+ thread_list.append(Thread(target=kronos_file, kwargs={'once': once,
502
+ 'sleep_time': sleep_time_files,
503
+ 'dataset_queue': dataset_queue}))
504
+ thread_list.append(Thread(target=kronos_dataset, kwargs={'once': once,
505
+ 'sleep_time': sleep_time_datasets,
506
+ 'dataset_queue': dataset_queue}))
507
+
508
+ [thread.start() for thread in thread_list]
509
+
510
+ logging.info('waiting for interrupts')
511
+
512
+ while len(thread_list) > 0:
513
+ thread_list = [thread.join(timeout=3) for thread in thread_list if thread and thread.is_alive()]
@@ -0,0 +1,14 @@
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.