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,530 @@
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
+ Methods common to different conveyor submitter daemons.
18
+ """
19
+
20
+ import datetime
21
+ import functools
22
+ import itertools
23
+ import logging
24
+ import re
25
+ from typing import TYPE_CHECKING
26
+
27
+ from rucio.common.config import config_get_bool
28
+ from rucio.common.exception import (InvalidRSEExpression, TransferToolTimeout, TransferToolWrongAnswer, RequestNotFound,
29
+ DuplicateFileTransferSubmission, VONotFound, DatabaseException)
30
+ from rucio.common.stopwatch import Stopwatch
31
+ from rucio.core import request as request_core, transfer as transfer_core
32
+ from rucio.core.monitor import MetricManager
33
+ from rucio.core.replica import add_replicas, tombstone_from_delay, update_replica_state
34
+ from rucio.core.request import set_request_state, queue_requests
35
+ from rucio.core.rse import list_rses
36
+ from rucio.core.rse_expression_parser import parse_expression
37
+ from rucio.core.transfer import build_transfer_paths
38
+ from rucio.core.vo import list_vos
39
+ from rucio.db.sqla import models
40
+ from rucio.db.sqla.constants import RequestState, ReplicaState
41
+ from rucio.db.sqla.session import transactional_session
42
+ from rucio.rse import rsemanager as rsemgr
43
+
44
+ if TYPE_CHECKING:
45
+ from collections.abc import Callable, Sequence
46
+ from typing import Optional
47
+ from rucio.core.transfer import DirectTransferDefinition
48
+ from rucio.transfertool.transfertool import TransferToolBuilder
49
+ from sqlalchemy.orm import Session
50
+
51
+ METRICS = MetricManager(module=__name__)
52
+
53
+
54
+ def pick_and_prepare_submission_path(requests_with_sources, topology, protocol_factory, default_tombstone_delay=transfer_core.DEFAULT_MULTIHOP_TOMBSTONE_DELAY,
55
+ admin_accounts=None, schemes=None, failover_schemes=None, max_sources=4, transfertools=None, logger=logging.log):
56
+ """
57
+ For each transfer, pick a (sub)path; and a transfertool to be used to submit that (sub)path
58
+ """
59
+
60
+ # for each source, compute the (possibly multihop) path between it and the transfer destination
61
+ _build_paths_fnc = functools.partial(
62
+ build_transfer_paths,
63
+ topology=topology,
64
+ protocol_factory=protocol_factory,
65
+ schemes=schemes,
66
+ failover_schemes=failover_schemes,
67
+ max_sources=max_sources,
68
+ admin_accounts=admin_accounts,
69
+ transfertools=transfertools,
70
+ logger=logger,
71
+ )
72
+
73
+ # Assign paths to be executed by transfertools
74
+ # if the chosen best path is a multihop, create intermediate replicas and the intermediate transfer requests
75
+ _assign_path_fnc = functools.partial(
76
+ assign_paths_to_transfertool_and_create_hops,
77
+ default_tombstone_delay=default_tombstone_delay,
78
+ transfertools=transfertools,
79
+ logger=logger,
80
+ )
81
+
82
+ # For requests which had source_rse_id set in the database, start by verifying that it's possible to
83
+ # use this source to submit it. If it's possible, use it, otherwise, fallback to searching all
84
+ # possible sources for the request
85
+ requests_to_load_partially = {}
86
+ requests_to_load_fully = {}
87
+ for request_id, rws in requests_with_sources.items():
88
+ if rws.requested_source and datetime.datetime.utcnow() - datetime.timedelta(hours=6) < rws.requested_at:
89
+ requests_to_load_partially[request_id] = rws
90
+ else:
91
+ requests_to_load_fully[request_id] = rws
92
+
93
+ paths_by_transfertool_builder_partial = {}
94
+ if requests_to_load_partially:
95
+ candidate_paths, reqs_no_source, reqs_scheme_mismatch, reqs_only_tape_source, reqs_unsupported_transfertool = _build_paths_fnc(
96
+ requests_with_sources=list(requests_to_load_partially.values()),
97
+ requested_source_only=True,
98
+ )
99
+ paths_by_transfertool_builder_partial, reqs_no_host = _assign_path_fnc(
100
+ candidate_paths_by_request_id=candidate_paths,
101
+ )
102
+ for request_id in itertools.chain(reqs_no_host, reqs_no_source, reqs_scheme_mismatch, reqs_only_tape_source, reqs_unsupported_transfertool):
103
+ rws = requests_to_load_partially[request_id]
104
+ logger(logging.INFO, "%s: source_rse_id was set to %s, but cannot be used. Fall-back to full source scan.", request_id, str(rws.requested_source))
105
+ requests_to_load_fully[request_id] = rws
106
+
107
+ candidate_paths, reqs_no_source, reqs_scheme_mismatch, reqs_only_tape_source, reqs_unsupported_transfertool = _build_paths_fnc(
108
+ requests_with_sources=list(requests_to_load_fully.values())
109
+ )
110
+ paths_by_transfertool_builder, reqs_no_host = _assign_path_fnc(
111
+ candidate_paths_by_request_id=candidate_paths
112
+ )
113
+ # Merge the two dicts with submission work into one
114
+ for tt_builder, paths in paths_by_transfertool_builder_partial.items():
115
+ paths_by_transfertool_builder.setdefault(tt_builder, []).extend(paths)
116
+
117
+ if reqs_unsupported_transfertool:
118
+ logger(logging.INFO, "Ignoring request because of unsupported transfertool: %s", reqs_unsupported_transfertool)
119
+ reqs_no_source.update(reqs_no_host)
120
+ if reqs_no_source:
121
+ logger(logging.INFO, "Marking requests as no-sources: %s", reqs_no_source)
122
+ request_core.set_requests_state_if_possible(reqs_no_source, RequestState.NO_SOURCES, logger=logger)
123
+ if reqs_only_tape_source:
124
+ logger(logging.INFO, "Marking requests as only-tape-sources: %s", reqs_only_tape_source)
125
+ request_core.set_requests_state_if_possible(reqs_only_tape_source, RequestState.ONLY_TAPE_SOURCES, logger=logger)
126
+ if reqs_scheme_mismatch:
127
+ logger(logging.INFO, "Marking requests as scheme-mismatch: %s", reqs_scheme_mismatch)
128
+ request_core.set_requests_state_if_possible(reqs_scheme_mismatch, RequestState.MISMATCH_SCHEME, logger=logger)
129
+
130
+ return paths_by_transfertool_builder
131
+
132
+
133
+ def __assign_to_transfertool(
134
+ transfer_path: "list[DirectTransferDefinition]",
135
+ transfertools: "Optional[list[str]]",
136
+ logger: "Callable",
137
+ ) -> "list[tuple[list[DirectTransferDefinition], Optional[TransferToolBuilder]]]":
138
+ """
139
+ Iterate over a multihop path and assign sub-paths to transfertools in chucks from left to right.
140
+
141
+ Assignment is done in a greedy way. At each step, the first transfertool which can submit any non-empty prefix
142
+ is selected. No backtracking is done to find better alternatives.
143
+
144
+ For example, for a path A->B->C->D->E, A->B->C may be assigned to transfertool1; while C->D->E to transfertool2.
145
+ This even if transfertool2 could submit the full path in one step without splitting it.
146
+ """
147
+ if transfertools is None:
148
+ return [(transfer_path, None)]
149
+
150
+ remaining_hops = transfer_path
151
+ tt_builder_for_hops = []
152
+ while remaining_hops:
153
+ tt_builder = None
154
+ assigned_hops = []
155
+ for tt_name in transfertools:
156
+ transfertool_cls = transfer_core.TRANSFERTOOL_CLASSES_BY_NAME.get(tt_name)
157
+ if not transfertool_cls:
158
+ continue
159
+ assigned_hops, tt_builder = transfertool_cls.submission_builder_for_path(remaining_hops, logger=logger)
160
+ if assigned_hops:
161
+ break
162
+
163
+ if not assigned_hops:
164
+ break
165
+
166
+ remaining_hops = remaining_hops[len(assigned_hops):]
167
+ tt_builder_for_hops.append((assigned_hops, tt_builder))
168
+
169
+ if remaining_hops:
170
+ # We cannot submit the whole path
171
+ return []
172
+
173
+ return tt_builder_for_hops
174
+
175
+
176
+ def assign_paths_to_transfertool_and_create_hops(
177
+ candidate_paths_by_request_id: "dict[str: list[DirectTransferDefinition]]",
178
+ default_tombstone_delay: int,
179
+ transfertools: "Optional[list[str]]" = None,
180
+ logger: "Callable" = logging.log,
181
+ ) -> "tuple[dict[TransferToolBuilder, list[DirectTransferDefinition]], set[str]]":
182
+ """
183
+ for each request, pick the first path which can be submitted by one of the transfertools.
184
+ If the chosen path is multihop, create all missing intermediate requests and replicas.
185
+ """
186
+ reqs_no_host = set()
187
+ paths_by_transfertool_builder = {}
188
+ for request_id, candidate_paths in candidate_paths_by_request_id.items():
189
+ try:
190
+ hops_to_submit, builder_to_use = __assign_paths_to_transfertool_and_create_hops(
191
+ request_id,
192
+ candidate_paths,
193
+ default_tombstone_delay=default_tombstone_delay,
194
+ transfertools=transfertools,
195
+ logger=logger,
196
+ )
197
+ except DatabaseException as e:
198
+ if re.match('.*SOURCES_(REQ_ID|REPLICA|RSES|DEST_RSES)_FK.*', str(e.args[0])):
199
+ logger(logging.WARNING, '%s: Cannot sync DB sources. Skip the transfer for now.', request_id)
200
+ continue
201
+ raise
202
+
203
+ if hops_to_submit is None:
204
+ reqs_no_host.add(request_id)
205
+ continue
206
+
207
+ if not hops_to_submit:
208
+ continue
209
+
210
+ paths_by_transfertool_builder.setdefault(builder_to_use, []).append(hops_to_submit)
211
+ return paths_by_transfertool_builder, reqs_no_host
212
+
213
+
214
+ @transactional_session
215
+ def __assign_paths_to_transfertool_and_create_hops(
216
+ request_id: str,
217
+ candidate_paths: "Sequence[list[DirectTransferDefinition]]",
218
+ default_tombstone_delay: int,
219
+ transfertools: "Optional[list[str]]" = None,
220
+ *,
221
+ logger: "Callable" = logging.log,
222
+ session: "Session",
223
+ ) -> "tuple[Optional[list[DirectTransferDefinition]], Optional[TransferToolBuilder]]":
224
+ """
225
+ Out of a sequence of candidate paths for the given request, pick the first path which can
226
+ be submitted by one of the transfertools.
227
+ If the chosen path is multihop, create all missing intermediate requests and replicas.
228
+ """
229
+ # Get the rws object from any candidate path. It is the same for all candidate paths. For multihop, the initial request is the last hop
230
+ rws = candidate_paths[0][-1].rws
231
+
232
+ # Selects the first path which can be submitted using a chain of supported transfertools
233
+ # and for which the creation of intermediate hops (if it is a multihop) works correctly
234
+ best_path = None
235
+ builder_to_use = None
236
+ hops_to_submit = None
237
+ must_skip_submission = False
238
+
239
+ tt_assignments = [(transfer_path, __assign_to_transfertool(transfer_path, transfertools, logger=logger))
240
+ for transfer_path in candidate_paths]
241
+ # Prioritize the paths which need less transfertool transitions.
242
+ # Ideally, the entire path should be submitted to a single transfertool
243
+ for transfer_path, tt_assignment in sorted(tt_assignments, key=lambda t: len(t[1])):
244
+ if not tt_assignment:
245
+ logger(logging.INFO, '%s: None of the transfertools can submit the request: %s', request_id, transfertools)
246
+ continue
247
+
248
+ # Set the 'transfertool' field on the intermediate hops which should be created in the database
249
+ for sub_path, tt_builder in tt_assignment:
250
+ if tt_builder:
251
+ for hop in sub_path:
252
+ if hop is not transfer_path[-1]:
253
+ hop.rws.transfertool = tt_builder.transfertool_class.external_name
254
+ created, must_skip_submission = __create_missing_replicas_and_requests(
255
+ transfer_path, default_tombstone_delay, logger=logger, session=session
256
+ )
257
+ if created:
258
+ best_path = transfer_path
259
+ # Only the first sub-path will be submitted to the corresponding transfertool,
260
+ # the rest of the hops will wait for first hops to be transferred
261
+ hops_to_submit, builder_to_use = tt_assignment[0]
262
+ if created or must_skip_submission:
263
+ break
264
+
265
+ if builder_to_use is None and hops_to_submit is None:
266
+ logger(logging.INFO, '%s: Cannot pick transfertool, or create intermediate requests' % request_id)
267
+ return hops_to_submit, builder_to_use
268
+
269
+ transfer_core.ensure_db_sources(best_path, logger=logger, session=session)
270
+
271
+ if len(best_path) > 1:
272
+ logger(logging.INFO, '%s: Best path is multihop: %s' % (rws.request_id, transfer_core.transfer_path_str(best_path)))
273
+ elif best_path is not candidate_paths[0] or len(best_path[0].sources) > 1:
274
+ # Only print singlehop if it brings additional information:
275
+ # - either it's not the first candidate path
276
+ # - or it's a multi-source
277
+ # in other cases, it doesn't bring any additional information to what is known from previous logs
278
+ logger(logging.INFO, '%s: Best path is direct: %s' % (rws.request_id, transfer_core.transfer_path_str(best_path)))
279
+
280
+ if must_skip_submission:
281
+ logger(logging.INFO, '%s: Part of the transfer is already being handled. Skip for now.' % request_id)
282
+ hops_to_submit = []
283
+ return hops_to_submit, builder_to_use
284
+
285
+ if len(hops_to_submit) < len(best_path):
286
+ logger(logging.INFO, '%s: Only first %d hops will be submitted by %s', request_id, len(hops_to_submit), builder_to_use)
287
+
288
+ return hops_to_submit, builder_to_use
289
+
290
+
291
+ @transactional_session
292
+ def __create_missing_replicas_and_requests(
293
+ transfer_path: "list[DirectTransferDefinition]",
294
+ default_tombstone_delay: int,
295
+ *,
296
+ logger: "Callable",
297
+ session: "Session"
298
+ ) -> tuple[bool, bool]:
299
+ """
300
+ Create replicas and requests in the database for the intermediate hops
301
+ """
302
+ initial_request_id = transfer_path[-1].rws.request_id
303
+ creation_successful = True
304
+ must_skip_submission = False
305
+ # Iterate the path in reverse order. The last hop is the initial request, so
306
+ # next_hop.rws.request_id will always be initialized when handling the current hop.
307
+ for i in reversed(range(len(transfer_path))):
308
+ hop = transfer_path[i]
309
+ rws = hop.rws
310
+ if rws.request_id:
311
+ continue
312
+
313
+ tombstone_delay = rws.dest_rse.attributes.get('multihop_tombstone_delay', default_tombstone_delay)
314
+ try:
315
+ tombstone = tombstone_from_delay(tombstone_delay)
316
+ except ValueError:
317
+ logger(logging.ERROR, "%s: Cannot parse multihop tombstone delay %s", initial_request_id, tombstone_delay)
318
+ creation_successful = False
319
+ break
320
+
321
+ files = [{'scope': rws.scope,
322
+ 'name': rws.name,
323
+ 'bytes': rws.byte_count,
324
+ 'adler32': rws.adler32,
325
+ 'md5': rws.md5,
326
+ 'tombstone': tombstone,
327
+ 'state': 'C'}]
328
+ try:
329
+ add_replicas(rse_id=rws.dest_rse.id,
330
+ files=files,
331
+ account=rws.account,
332
+ ignore_availability=False,
333
+ dataset_meta=None,
334
+ session=session)
335
+ # Set replica state to Copying in case replica already existed in another state.
336
+ # Can happen when a multihop transfer failed previously, and we are re-scheduling it now.
337
+ update_replica_state(rse_id=rws.dest_rse.id, scope=rws.scope, name=rws.name, state=ReplicaState.COPYING, session=session)
338
+ except Exception as error:
339
+ logger(logging.ERROR, '%s: Problem adding replicas on %s : %s', initial_request_id, rws.dest_rse, str(error))
340
+
341
+ rws.attributes['is_intermediate_hop'] = True
342
+ rws.attributes['source_replica_expression'] = hop.src.rse.name
343
+ req_to_queue = {'dest_rse_id': rws.dest_rse.id,
344
+ 'source_rse_id': hop.src.rse.id,
345
+ 'state': RequestState.QUEUED,
346
+ 'scope': rws.scope,
347
+ 'name': rws.name,
348
+ 'rule_id': '00000000000000000000000000000000', # Dummy Rule ID used for multihop. TODO: Replace with actual rule_id once we can flag intermediate requests
349
+ 'attributes': rws.attributes,
350
+ 'request_type': rws.request_type,
351
+ 'retry_count': rws.retry_count,
352
+ 'account': rws.account,
353
+ 'requested_at': datetime.datetime.utcnow()}
354
+ if rws.transfertool:
355
+ req_to_queue['transfertool'] = rws.transfertool
356
+ new_req = queue_requests(requests=[req_to_queue], session=session)
357
+ # If a request already exists, new_req will be an empty list.
358
+ if new_req:
359
+ db_req = new_req[0]
360
+ logger(logging.DEBUG, '%s: New request created for the transfer between %s and %s : %s', initial_request_id, transfer_path[0].src, transfer_path[-1].dst, db_req['id'])
361
+ else:
362
+ db_req = request_core.get_request_by_did(rws.scope, rws.name, rws.dest_rse.id, session=session)
363
+ # A transfer already exists for part of the path. Just construct the remaining
364
+ # path, but don't submit the transfer. We must wait for the existing transfer to be
365
+ # completed before continuing.
366
+ must_skip_submission = True
367
+ logger(logging.DEBUG, '%s: Reusing intermediate hop between %s and %s : %s', initial_request_id, transfer_path[0].src, transfer_path[-1].dst, db_req['id'])
368
+
369
+ models.TransferHop(request_id=db_req['id'],
370
+ next_hop_request_id=transfer_path[i + 1].rws.request_id,
371
+ initial_request_id=initial_request_id,
372
+ ).save(session=session, flush=False)
373
+ rws.request_id = db_req['id']
374
+ rws.requested_at = db_req['requested_at']
375
+
376
+ return creation_successful, must_skip_submission
377
+
378
+
379
+ def submit_transfer(transfertool_obj, transfers, job_params, timeout=None, logger=logging.log):
380
+ """
381
+ Submit a transfer or staging request
382
+
383
+ :param transfertool_obj: The transfertool object to be used for submission
384
+ :param transfers: Transfer objects to be submitted
385
+ :param job_params: Parameters to be used for all transfers in the given job.
386
+ :param submitter: Name of the submitting entity.
387
+ :param timeout: Timeout
388
+ :param logger: Optional decorated logger that can be passed from the calling daemons or servers.
389
+ """
390
+
391
+ for transfer in transfers:
392
+ try:
393
+ transfer_core.mark_submitting(transfer, external_host=transfertool_obj.external_host, logger=logger)
394
+ except RequestNotFound as error:
395
+ logger(logging.ERROR, str(error))
396
+ return
397
+ except Exception:
398
+ logger(logging.ERROR, 'Failed to prepare requests %s state to SUBMITTING. Mark it SUBMISSION_FAILED and abort submission.' % [str(t.rws) for t in transfers], exc_info=True)
399
+ set_request_state(request_id=transfer.rws.request_id, state=RequestState.SUBMISSION_FAILED)
400
+ return
401
+
402
+ try:
403
+ _submit_transfers(transfertool_obj, transfers, job_params, timeout, logger)
404
+ except DuplicateFileTransferSubmission as error:
405
+ logger(logging.WARNING, 'Failed to bulk submit a job because of duplicate file : %s', str(error))
406
+ logger(logging.INFO, 'Submitting files one by one')
407
+ for transfer in transfers:
408
+ _submit_transfers(transfertool_obj, [transfer], job_params, timeout, logger)
409
+
410
+
411
+ def _submit_transfers(transfertool_obj, transfers, job_params, timeout=None, logger=logging.log):
412
+ """
413
+ helper function for submit_transfers. Performs the actual submission of one or more transfers.
414
+
415
+ If the bulk submission of multiple transfers fails due to duplicate submissions, the exception
416
+ is propagated to the caller context, which is then responsible for calling this function again for each
417
+ of the transfers separately.
418
+ """
419
+ logger(logging.DEBUG, 'About to submit job to %s with timeout %s' % (transfertool_obj, timeout))
420
+ # A eid is returned if the job is properly submitted otherwise an exception is raised
421
+ is_bulk = len(transfers) > 1
422
+ eid = None
423
+ stopwatch = Stopwatch()
424
+ state_to_set = RequestState.SUBMISSION_FAILED
425
+ try:
426
+ METRICS.counter('submit_transfer').inc()
427
+ eid = transfertool_obj.submit(transfers, job_params, timeout)
428
+ state_to_set = RequestState.SUBMITTED
429
+ except DuplicateFileTransferSubmission:
430
+ if is_bulk:
431
+ raise
432
+ except (TransferToolTimeout, TransferToolWrongAnswer) as error:
433
+ logger(logging.ERROR, 'Failed to submit a job with error %s', str(error), exc_info=True)
434
+ except Exception as error:
435
+ logger(logging.ERROR, 'Failed to submit a job with error %s', str(error), exc_info=True)
436
+ # Keep the behavior from before the refactoring: in case of unexpected exception, only
437
+ # update request state on individual transfers, and do nothing for bulks.
438
+ # Don't know why it's like that.
439
+ #
440
+ # FIXME: shouldn't we always set the state to SUBMISSION_FAILED?
441
+ if is_bulk:
442
+ state_to_set = None
443
+
444
+ if eid is not None:
445
+ stopwatch.stop()
446
+ logger(logging.DEBUG, 'Submit job %s to %s in %s seconds' % (eid, transfertool_obj, stopwatch.elapsed))
447
+ METRICS.timer('submit_bulk_transfer_per_file').observe(stopwatch.elapsed / (len(transfers) or 1))
448
+ METRICS.counter('submit_bulk_transfer').inc(len(transfers))
449
+
450
+ if state_to_set:
451
+ try:
452
+ transfer_core.set_transfers_state(
453
+ transfers,
454
+ state=state_to_set,
455
+ external_host=transfertool_obj.external_host,
456
+ external_id=eid,
457
+ submitted_at=datetime.datetime.utcnow(),
458
+ transfertool=transfertool_obj.external_name,
459
+ logger=logger
460
+ )
461
+ except Exception:
462
+ logger(logging.ERROR, 'Failed to register transfer state with error', exc_info=True)
463
+ if eid is not None:
464
+ # The job is still submitted in the file transfer service but the request is not updated.
465
+ # Possibility to have a double submission during the next cycle. Try to cancel the external request.
466
+ try:
467
+ logger(logging.INFO, 'Cancel transfer %s on %s', eid, transfertool_obj)
468
+ transfer_core.cancel_transfer(transfertool_obj, eid)
469
+ except Exception:
470
+ logger(logging.ERROR, 'Failed to cancel transfers %s on %s with error' % (eid, transfertool_obj), exc_info=True)
471
+
472
+
473
+ def get_conveyor_rses(rses=None, include_rses=None, exclude_rses=None, vos=None, logger=logging.log):
474
+ """
475
+ Get a list of rses for conveyor
476
+
477
+ :param rses: List of rses (Single-VO only)
478
+ :param include_rses: RSEs to include
479
+ :param exclude_rses: RSEs to exclude
480
+ :param vos: VOs on which to look for RSEs. Only used in multi-VO mode.
481
+ If None, we either use all VOs if run from "def", or the current VO otherwise.
482
+ :param logger: Optional decorated logger that can be passed from the calling daemons or servers.
483
+ :return: List of working rses
484
+ """
485
+ multi_vo = config_get_bool('common', 'multi_vo', raise_exception=False, default=False)
486
+ if not multi_vo:
487
+ if vos:
488
+ logger(logging.WARNING, 'Ignoring argument vos, this is only applicable in a multi-VO setup.')
489
+ vos = ['def']
490
+ else:
491
+ if vos:
492
+ invalid = set(vos) - set([v['vo'] for v in list_vos()])
493
+ if invalid:
494
+ msg = 'VO{} {} cannot be found'.format('s' if len(invalid) > 1 else '', ', '.join([repr(v) for v in invalid]))
495
+ raise VONotFound(msg)
496
+ else:
497
+ vos = [v['vo'] for v in list_vos()]
498
+ logger(logging.INFO, 'This instance will work on VO%s: %s' % ('s' if len(vos) > 1 else '', ', '.join([v for v in vos])))
499
+
500
+ working_rses = []
501
+ rses_list = []
502
+ for vo in vos:
503
+ rses_list.extend(list_rses(filters={'vo': vo}))
504
+ if rses:
505
+ working_rses = [rse for rse in rses_list if rse['rse'] in rses]
506
+
507
+ if include_rses:
508
+ for vo in vos:
509
+ try:
510
+ parsed_rses = parse_expression(include_rses, filter_={'vo': vo}, session=None)
511
+ except InvalidRSEExpression:
512
+ logger(logging.ERROR, "Invalid RSE exception %s to include RSEs", include_rses)
513
+ else:
514
+ for rse in parsed_rses:
515
+ if rse not in working_rses:
516
+ working_rses.append(rse)
517
+
518
+ if not (rses or include_rses):
519
+ working_rses = rses_list
520
+
521
+ if exclude_rses:
522
+ try:
523
+ parsed_rses = parse_expression(exclude_rses, session=None)
524
+ except InvalidRSEExpression as error:
525
+ logger(logging.ERROR, "Invalid RSE exception %s to exclude RSEs: %s", exclude_rses, error)
526
+ else:
527
+ working_rses = [rse for rse in working_rses if rse not in parsed_rses]
528
+
529
+ working_rses = [rsemgr.get_rse_info(rse_id=rse['id']) for rse in working_rses]
530
+ return working_rses