rucio 35.7.0__py3-none-any.whl → 37.0.0rc2__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 (268) hide show
  1. rucio/alembicrevision.py +1 -1
  2. rucio/{daemons/c3po/collectors → cli}/__init__.py +1 -0
  3. rucio/cli/account.py +216 -0
  4. rucio-35.7.0.data/scripts/rucio → rucio/cli/bin_legacy/rucio.py +769 -486
  5. rucio-35.7.0.data/scripts/rucio-admin → rucio/cli/bin_legacy/rucio_admin.py +476 -423
  6. rucio/cli/command.py +272 -0
  7. rucio/cli/config.py +72 -0
  8. rucio/cli/did.py +191 -0
  9. rucio/cli/download.py +128 -0
  10. rucio/cli/lifetime_exception.py +33 -0
  11. rucio/cli/replica.py +162 -0
  12. rucio/cli/rse.py +293 -0
  13. rucio/cli/rule.py +158 -0
  14. rucio/cli/scope.py +40 -0
  15. rucio/cli/subscription.py +73 -0
  16. rucio/cli/upload.py +60 -0
  17. rucio/cli/utils.py +226 -0
  18. rucio/client/accountclient.py +0 -1
  19. rucio/client/baseclient.py +33 -24
  20. rucio/client/client.py +45 -1
  21. rucio/client/didclient.py +5 -3
  22. rucio/client/downloadclient.py +6 -8
  23. rucio/client/replicaclient.py +0 -2
  24. rucio/client/richclient.py +317 -0
  25. rucio/client/rseclient.py +4 -4
  26. rucio/client/uploadclient.py +26 -12
  27. rucio/common/bittorrent.py +234 -0
  28. rucio/common/cache.py +66 -29
  29. rucio/common/checksum.py +168 -0
  30. rucio/common/client.py +122 -0
  31. rucio/common/config.py +22 -35
  32. rucio/common/constants.py +61 -3
  33. rucio/common/didtype.py +72 -24
  34. rucio/common/dumper/__init__.py +45 -38
  35. rucio/common/dumper/consistency.py +75 -30
  36. rucio/common/dumper/data_models.py +63 -19
  37. rucio/common/dumper/path_parsing.py +19 -8
  38. rucio/common/exception.py +65 -8
  39. rucio/common/extra.py +5 -10
  40. rucio/common/logging.py +13 -13
  41. rucio/common/pcache.py +8 -7
  42. rucio/common/plugins.py +59 -27
  43. rucio/common/policy.py +12 -3
  44. rucio/common/schema/__init__.py +84 -34
  45. rucio/common/schema/generic.py +0 -17
  46. rucio/common/schema/generic_multi_vo.py +0 -17
  47. rucio/common/stomp_utils.py +383 -119
  48. rucio/common/test_rucio_server.py +12 -6
  49. rucio/common/types.py +132 -52
  50. rucio/common/utils.py +93 -643
  51. rucio/core/account_limit.py +14 -12
  52. rucio/core/authentication.py +2 -2
  53. rucio/core/config.py +23 -42
  54. rucio/core/credential.py +14 -15
  55. rucio/core/did.py +5 -1
  56. rucio/core/did_meta_plugins/elasticsearch_meta.py +407 -0
  57. rucio/core/did_meta_plugins/filter_engine.py +62 -3
  58. rucio/core/did_meta_plugins/json_meta.py +2 -2
  59. rucio/core/did_meta_plugins/mongo_meta.py +43 -30
  60. rucio/core/did_meta_plugins/postgres_meta.py +75 -39
  61. rucio/core/identity.py +6 -5
  62. rucio/core/importer.py +4 -3
  63. rucio/core/lifetime_exception.py +2 -2
  64. rucio/core/lock.py +8 -7
  65. rucio/core/message.py +6 -0
  66. rucio/core/monitor.py +30 -29
  67. rucio/core/naming_convention.py +2 -2
  68. rucio/core/nongrid_trace.py +2 -2
  69. rucio/core/oidc.py +11 -9
  70. rucio/core/permission/__init__.py +79 -37
  71. rucio/core/permission/generic.py +1 -7
  72. rucio/core/permission/generic_multi_vo.py +1 -7
  73. rucio/core/quarantined_replica.py +4 -3
  74. rucio/core/replica.py +464 -139
  75. rucio/core/replica_sorter.py +55 -59
  76. rucio/core/request.py +34 -32
  77. rucio/core/rse.py +301 -97
  78. rucio/core/rse_counter.py +1 -2
  79. rucio/core/rse_expression_parser.py +7 -7
  80. rucio/core/rse_selector.py +9 -7
  81. rucio/core/rule.py +41 -40
  82. rucio/core/rule_grouping.py +42 -40
  83. rucio/core/scope.py +5 -4
  84. rucio/core/subscription.py +26 -28
  85. rucio/core/topology.py +11 -11
  86. rucio/core/trace.py +2 -2
  87. rucio/core/transfer.py +29 -15
  88. rucio/core/volatile_replica.py +4 -3
  89. rucio/daemons/atropos/atropos.py +1 -1
  90. rucio/daemons/auditor/__init__.py +2 -2
  91. rucio/daemons/auditor/srmdumps.py +6 -6
  92. rucio/daemons/automatix/automatix.py +32 -21
  93. rucio/daemons/badreplicas/necromancer.py +2 -2
  94. rucio/daemons/bb8/nuclei_background_rebalance.py +1 -1
  95. rucio/daemons/bb8/t2_background_rebalance.py +1 -1
  96. rucio/daemons/cache/consumer.py +26 -90
  97. rucio/daemons/common.py +15 -25
  98. rucio/daemons/conveyor/finisher.py +2 -2
  99. rucio/daemons/conveyor/poller.py +18 -28
  100. rucio/daemons/conveyor/receiver.py +53 -123
  101. rucio/daemons/conveyor/stager.py +1 -0
  102. rucio/daemons/conveyor/submitter.py +3 -3
  103. rucio/daemons/hermes/hermes.py +129 -369
  104. rucio/daemons/judge/evaluator.py +2 -2
  105. rucio/daemons/oauthmanager/oauthmanager.py +3 -3
  106. rucio/daemons/reaper/dark_reaper.py +7 -3
  107. rucio/daemons/reaper/reaper.py +12 -16
  108. rucio/daemons/rsedecommissioner/config.py +1 -1
  109. rucio/daemons/rsedecommissioner/profiles/generic.py +5 -4
  110. rucio/daemons/rsedecommissioner/profiles/types.py +7 -6
  111. rucio/daemons/rsedecommissioner/rse_decommissioner.py +1 -1
  112. rucio/daemons/storage/consistency/actions.py +8 -6
  113. rucio/daemons/tracer/kronos.py +117 -142
  114. rucio/db/sqla/constants.py +5 -0
  115. rucio/db/sqla/migrate_repo/versions/1677d4d803c8_split_rse_availability_into_multiple.py +4 -4
  116. rucio/db/sqla/migrate_repo/versions/30d5206e9cad_increase_oauthrequest_redirect_msg_.py +37 -0
  117. rucio/db/sqla/models.py +157 -154
  118. rucio/db/sqla/session.py +58 -27
  119. rucio/db/sqla/types.py +2 -2
  120. rucio/db/sqla/util.py +2 -2
  121. rucio/gateway/account.py +18 -12
  122. rucio/gateway/account_limit.py +137 -60
  123. rucio/gateway/authentication.py +18 -12
  124. rucio/gateway/config.py +30 -20
  125. rucio/gateway/credential.py +9 -10
  126. rucio/gateway/did.py +70 -53
  127. rucio/gateway/dirac.py +6 -4
  128. rucio/gateway/exporter.py +3 -2
  129. rucio/gateway/heartbeat.py +6 -4
  130. rucio/gateway/identity.py +36 -51
  131. rucio/gateway/importer.py +3 -2
  132. rucio/gateway/lifetime_exception.py +3 -2
  133. rucio/gateway/meta_conventions.py +17 -6
  134. rucio/gateway/permission.py +4 -1
  135. rucio/gateway/quarantined_replica.py +3 -2
  136. rucio/gateway/replica.py +31 -22
  137. rucio/gateway/request.py +27 -18
  138. rucio/gateway/rse.py +69 -37
  139. rucio/gateway/rule.py +46 -26
  140. rucio/gateway/scope.py +3 -2
  141. rucio/gateway/subscription.py +14 -11
  142. rucio/gateway/vo.py +12 -8
  143. rucio/rse/__init__.py +3 -3
  144. rucio/rse/protocols/bittorrent.py +11 -1
  145. rucio/rse/protocols/cache.py +0 -11
  146. rucio/rse/protocols/dummy.py +0 -11
  147. rucio/rse/protocols/gfal.py +14 -9
  148. rucio/rse/protocols/globus.py +1 -1
  149. rucio/rse/protocols/http_cache.py +1 -1
  150. rucio/rse/protocols/posix.py +2 -2
  151. rucio/rse/protocols/protocol.py +84 -317
  152. rucio/rse/protocols/rclone.py +2 -1
  153. rucio/rse/protocols/rfio.py +10 -1
  154. rucio/rse/protocols/ssh.py +2 -1
  155. rucio/rse/protocols/storm.py +2 -13
  156. rucio/rse/protocols/webdav.py +74 -30
  157. rucio/rse/protocols/xrootd.py +2 -1
  158. rucio/rse/rsemanager.py +170 -53
  159. rucio/rse/translation.py +260 -0
  160. rucio/tests/common.py +23 -13
  161. rucio/tests/common_server.py +26 -9
  162. rucio/transfertool/bittorrent.py +15 -14
  163. rucio/transfertool/bittorrent_driver.py +5 -7
  164. rucio/transfertool/bittorrent_driver_qbittorrent.py +9 -8
  165. rucio/transfertool/fts3.py +20 -16
  166. rucio/transfertool/mock.py +2 -3
  167. rucio/vcsversion.py +4 -4
  168. rucio/version.py +7 -0
  169. rucio/web/rest/flaskapi/v1/accounts.py +17 -3
  170. rucio/web/rest/flaskapi/v1/auth.py +5 -5
  171. rucio/web/rest/flaskapi/v1/credentials.py +3 -2
  172. rucio/web/rest/flaskapi/v1/dids.py +21 -15
  173. rucio/web/rest/flaskapi/v1/identities.py +33 -9
  174. rucio/web/rest/flaskapi/v1/redirect.py +5 -4
  175. rucio/web/rest/flaskapi/v1/replicas.py +12 -8
  176. rucio/web/rest/flaskapi/v1/rses.py +15 -4
  177. rucio/web/rest/flaskapi/v1/traces.py +56 -19
  178. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/alembic.ini.template +1 -1
  179. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/alembic_offline.ini.template +1 -1
  180. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/rucio.cfg.atlas.client.template +3 -2
  181. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/rucio.cfg.template +3 -19
  182. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/rucio_multi_vo.cfg.template +1 -18
  183. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/requirements.server.txt +97 -68
  184. rucio-37.0.0rc2.data/scripts/rucio +133 -0
  185. rucio-37.0.0rc2.data/scripts/rucio-admin +97 -0
  186. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-atropos +2 -2
  187. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-auditor +2 -1
  188. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-automatix +2 -2
  189. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-cache-client +17 -10
  190. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-conveyor-receiver +1 -0
  191. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-kronos +1 -0
  192. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-minos +2 -2
  193. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-minos-temporary-expiration +2 -2
  194. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-necromancer +2 -2
  195. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-reaper +6 -6
  196. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-transmogrifier +2 -2
  197. rucio-37.0.0rc2.dist-info/METADATA +92 -0
  198. {rucio-35.7.0.dist-info → rucio-37.0.0rc2.dist-info}/RECORD +239 -245
  199. {rucio-35.7.0.dist-info → rucio-37.0.0rc2.dist-info}/licenses/AUTHORS.rst +3 -0
  200. rucio/common/schema/atlas.py +0 -413
  201. rucio/common/schema/belleii.py +0 -408
  202. rucio/common/schema/domatpc.py +0 -401
  203. rucio/common/schema/escape.py +0 -426
  204. rucio/common/schema/icecube.py +0 -406
  205. rucio/core/permission/atlas.py +0 -1348
  206. rucio/core/permission/belleii.py +0 -1077
  207. rucio/core/permission/escape.py +0 -1078
  208. rucio/daemons/c3po/algorithms/__init__.py +0 -13
  209. rucio/daemons/c3po/algorithms/simple.py +0 -134
  210. rucio/daemons/c3po/algorithms/t2_free_space.py +0 -128
  211. rucio/daemons/c3po/algorithms/t2_free_space_only_pop.py +0 -130
  212. rucio/daemons/c3po/algorithms/t2_free_space_only_pop_with_network.py +0 -294
  213. rucio/daemons/c3po/c3po.py +0 -371
  214. rucio/daemons/c3po/collectors/agis.py +0 -108
  215. rucio/daemons/c3po/collectors/free_space.py +0 -81
  216. rucio/daemons/c3po/collectors/jedi_did.py +0 -57
  217. rucio/daemons/c3po/collectors/mock_did.py +0 -51
  218. rucio/daemons/c3po/collectors/network_metrics.py +0 -71
  219. rucio/daemons/c3po/collectors/workload.py +0 -112
  220. rucio/daemons/c3po/utils/__init__.py +0 -13
  221. rucio/daemons/c3po/utils/dataset_cache.py +0 -50
  222. rucio/daemons/c3po/utils/expiring_dataset_cache.py +0 -56
  223. rucio/daemons/c3po/utils/expiring_list.py +0 -62
  224. rucio/daemons/c3po/utils/popularity.py +0 -85
  225. rucio/daemons/c3po/utils/timeseries.py +0 -89
  226. rucio/rse/protocols/gsiftp.py +0 -92
  227. rucio-35.7.0.data/scripts/rucio-c3po +0 -85
  228. rucio-35.7.0.dist-info/METADATA +0 -72
  229. /rucio/{daemons/c3po → cli/bin_legacy}/__init__.py +0 -0
  230. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/globus-config.yml.template +0 -0
  231. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/ldap.cfg.template +0 -0
  232. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/mail_templates/rule_approval_request.tmpl +0 -0
  233. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/mail_templates/rule_approved_admin.tmpl +0 -0
  234. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/mail_templates/rule_approved_user.tmpl +0 -0
  235. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/mail_templates/rule_denied_admin.tmpl +0 -0
  236. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/mail_templates/rule_denied_user.tmpl +0 -0
  237. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/mail_templates/rule_ok_notification.tmpl +0 -0
  238. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/etc/rse-accounts.cfg.template +0 -0
  239. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/tools/bootstrap.py +0 -0
  240. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/tools/merge_rucio_configs.py +0 -0
  241. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/data/rucio/tools/reset_database.py +0 -0
  242. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-abacus-account +0 -0
  243. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-abacus-collection-replica +0 -0
  244. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-abacus-rse +0 -0
  245. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-bb8 +0 -0
  246. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-cache-consumer +0 -0
  247. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-conveyor-finisher +0 -0
  248. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-conveyor-poller +0 -0
  249. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-conveyor-preparer +0 -0
  250. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-conveyor-stager +0 -0
  251. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-conveyor-submitter +0 -0
  252. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-conveyor-throttler +0 -0
  253. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-dark-reaper +0 -0
  254. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-dumper +0 -0
  255. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-follower +0 -0
  256. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-hermes +0 -0
  257. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-judge-cleaner +0 -0
  258. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-judge-evaluator +0 -0
  259. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-judge-injector +0 -0
  260. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-judge-repairer +0 -0
  261. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-oauth-manager +0 -0
  262. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-replica-recoverer +0 -0
  263. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-rse-decommissioner +0 -0
  264. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-storage-consistency-actions +0 -0
  265. {rucio-35.7.0.data → rucio-37.0.0rc2.data}/scripts/rucio-undertaker +0 -0
  266. {rucio-35.7.0.dist-info → rucio-37.0.0rc2.dist-info}/WHEEL +0 -0
  267. {rucio-35.7.0.dist-info → rucio-37.0.0rc2.dist-info}/licenses/LICENSE +0 -0
  268. {rucio-35.7.0.dist-info → rucio-37.0.0rc2.dist-info}/top_level.txt +0 -0
@@ -1,13 +0,0 @@
1
- # Copyright European Organization for Nuclear Research (CERN) since 2012
2
- #
3
- # Licensed under the Apache License, Version 2.0 (the "License");
4
- # you may not use this file except in compliance with the License.
5
- # You may obtain a copy of the License at
6
- #
7
- # http://www.apache.org/licenses/LICENSE-2.0
8
- #
9
- # Unless required by applicable law or agreed to in writing, software
10
- # distributed under the License is distributed on an "AS IS" BASIS,
11
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
- # See the License for the specific language governing permissions and
13
- # limitations under the License.
@@ -1,134 +0,0 @@
1
- # Copyright European Organization for Nuclear Research (CERN) since 2012
2
- #
3
- # Licensed under the Apache License, Version 2.0 (the "License");
4
- # you may not use this file except in compliance with the License.
5
- # You may obtain a copy of the License at
6
- #
7
- # http://www.apache.org/licenses/LICENSE-2.0
8
- #
9
- # Unless required by applicable law or agreed to in writing, software
10
- # distributed under the License is distributed on an "AS IS" BASIS,
11
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
- # See the License for the specific language governing permissions and
13
- # limitations under the License.
14
-
15
- import logging
16
- from operator import itemgetter
17
- from random import shuffle
18
- from typing import TYPE_CHECKING, Any
19
-
20
- from rucio.common.exception import DataIdentifierNotFound
21
- from rucio.core.did import get_did
22
- from rucio.core.replica import list_dataset_replicas
23
- from rucio.daemons.c3po.collectors.agis import MappingCollector
24
- from rucio.daemons.c3po.collectors.workload import WorkloadCollector
25
- from rucio.db.sqla.constants import ReplicaState
26
-
27
- if TYPE_CHECKING:
28
- from rucio.common.types import InternalScope
29
-
30
-
31
- class PlacementAlgorithm:
32
- def __init__(self):
33
- self._mc = MappingCollector()
34
- self._wc = WorkloadCollector()
35
- self.__setup_penalties()
36
-
37
- def __setup_penalties(self) -> None:
38
- self._penalties = {}
39
- for panda_site in self._wc.get_sites():
40
- site = self._mc.panda_to_site(panda_site)
41
- self._penalties[site] = 0.1
42
-
43
- def __update_penalties(self) -> None:
44
- for site, penalty in self._penalties.items():
45
- if penalty > 0.1:
46
- self._penalties[site] = penalty - 0.1
47
-
48
- def place(self, did: tuple['InternalScope', str]) -> dict[str, Any]:
49
- self.__update_penalties()
50
- decision: dict[str, Any] = {'did': '{}:{}'.format(did['scope'].internal, did['name'])} # type: ignore (did is treated as a dict here, and as a tuple everywhere else)
51
- try:
52
- meta = get_did(did[0], did[1])
53
- except DataIdentifierNotFound:
54
- decision['error_reason'] = 'did does not exist'
55
- return decision
56
- if meta['length'] is None:
57
- meta['length'] = 0
58
- if meta['bytes'] is None:
59
- meta['bytes'] = 0
60
- logging.debug('got %s:%s, num_files: %d, bytes: %d' % (did[0], did[1], meta['length'], meta['bytes']))
61
-
62
- decision['length'] = meta['length']
63
- decision['bytes'] = meta['bytes']
64
-
65
- available_rses = []
66
- available_sites = []
67
- reps = list_dataset_replicas(did[0], did[1])
68
-
69
- num_reps = 0
70
- for rep in reps:
71
- if rep['state'] == ReplicaState.AVAILABLE:
72
- available_rses.append(rep['rse'])
73
- available_sites.append(self._mc.ddm_to_site(rep['rse']))
74
- num_reps += 1
75
-
76
- decision['replica_rses'] = available_rses
77
- decision['num_replicas'] = num_reps
78
- if num_reps >= 5:
79
- decision['error_reason'] = 'more than 4 replicas already exist'
80
- return decision
81
-
82
- site_ratios = {}
83
- site_job_info = {}
84
- for panda_site in self._wc.get_sites():
85
- site = self._mc.panda_to_site(panda_site)
86
- job_info = self._wc.get_job_info(panda_site)
87
- ratio = float(job_info[0]) / (float(job_info[1]) + float(job_info[2]) / 2)
88
- penalty = self._penalties[site]
89
- site_ratios[site] = ratio * penalty
90
- site_job_info[site] = (job_info, penalty)
91
-
92
- decision['site_ratios'] = site_ratios
93
- decision['site_job_info'] = site_job_info
94
- picked_site = None
95
- picked_rse = None
96
-
97
- for site, _ in sorted(site_ratios.items(), key=itemgetter(1)):
98
- if site in available_sites:
99
- continue
100
- rses_for_site = self._mc.site_to_ddm(site)
101
- if rses_for_site is None:
102
- continue
103
-
104
- for rse in rses_for_site:
105
- if 'DATADISK' in rse:
106
- picked_rse = rse
107
- picked_site = site
108
- break
109
- if picked_rse:
110
- break
111
-
112
- if picked_rse is None:
113
- decision['error_reason'] = 'could not pick RSE'
114
- return decision
115
-
116
- decision['destination_rse'] = picked_rse
117
- if picked_site:
118
- self._penalties[site] = 1
119
-
120
- picked_source = None
121
- shuffle(available_rses)
122
- for rse in available_rses:
123
- if 'TAPE' in rse:
124
- continue
125
- picked_source = rse
126
- break
127
-
128
- if picked_source is None:
129
- picked_source = available_rses[0]
130
-
131
- decision['source_rse'] = picked_source
132
- logging.debug("Picked %s as source and %s as destination RSE" % (picked_source, picked_rse))
133
-
134
- return decision
@@ -1,128 +0,0 @@
1
- # Copyright European Organization for Nuclear Research (CERN) since 2012
2
- #
3
- # Licensed under the Apache License, Version 2.0 (the "License");
4
- # you may not use this file except in compliance with the License.
5
- # You may obtain a copy of the License at
6
- #
7
- # http://www.apache.org/licenses/LICENSE-2.0
8
- #
9
- # Unless required by applicable law or agreed to in writing, software
10
- # distributed under the License is distributed on an "AS IS" BASIS,
11
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
- # See the License for the specific language governing permissions and
13
- # limitations under the License.
14
-
15
- import logging
16
- from operator import itemgetter
17
- from typing import TYPE_CHECKING, Any
18
-
19
- from rucio.common.config import config_get, config_get_int
20
- from rucio.common.exception import DataIdentifierNotFound
21
- from rucio.core.did import get_did
22
- from rucio.core.replica import list_dataset_replicas
23
- from rucio.core.rse import get_rse_name, list_rse_attributes
24
- from rucio.core.rse_expression_parser import parse_expression
25
- from rucio.daemons.c3po.collectors.free_space import FreeSpaceCollector
26
- from rucio.daemons.c3po.utils.dataset_cache import DatasetCache
27
- from rucio.daemons.c3po.utils.popularity import get_popularity
28
- from rucio.db.sqla.constants import ReplicaState
29
-
30
- if TYPE_CHECKING:
31
- from rucio.common.types import InternalScope
32
-
33
-
34
- class PlacementAlgorithm:
35
- """
36
- Placement algorithm that focusses on free space on T2 DATADISK RSEs.
37
- """
38
- def __init__(self):
39
- self._fsc = FreeSpaceCollector()
40
- self._dc = DatasetCache(config_get('c3po', 'redis_host'), config_get_int('c3po', 'redis_port'), timeout=86400)
41
-
42
- rse_expr = "tier=2&type=DATADISK"
43
- rse_attrs = parse_expression(rse_expr)
44
-
45
- self._rses = []
46
- for rse in rse_attrs:
47
- self._rses.append(rse['id'])
48
-
49
- self.__setup_penalties()
50
-
51
- def __setup_penalties(self) -> None:
52
- self._penalties = {}
53
- for rse_id in self._rses:
54
- self._penalties[rse_id] = 1.0
55
-
56
- def __update_penalties(self) -> None:
57
- for rse_id, penalty in self._penalties.items():
58
- if penalty > 1.0:
59
- self._penalties[rse_id] = penalty - 1
60
-
61
- def place(self, did: tuple['InternalScope', str]) -> dict[str, Any]:
62
- self.__update_penalties()
63
- decision: dict[str, Any] = {'did': '{}:{}'.format(did[0].internal, did[1])}
64
- if (did[0].external is not None) and (not did[0].external.startswith('data')) and (not did[0].external.startswith('mc')):
65
- decision['error_reason'] = 'not a data or mc dataset'
66
- return decision
67
-
68
- try:
69
- meta = get_did(did[0], did[1])
70
- except DataIdentifierNotFound:
71
- decision['error_reason'] = 'did does not exist'
72
- return decision
73
- if meta['length'] is None:
74
- meta['length'] = 0
75
- if meta['bytes'] is None:
76
- meta['bytes'] = 0
77
- logging.debug('got %s:%s, num_files: %d, bytes: %d' % (did[0], did[1], meta['length'], meta['bytes']))
78
-
79
- decision['length'] = meta['length']
80
- decision['bytes'] = meta['bytes']
81
-
82
- last_accesses = self._dc.get_did(did)
83
- self._dc.add_did(did)
84
-
85
- decision['last_accesses'] = last_accesses
86
-
87
- pop = get_popularity(did)
88
- decision['popularity'] = pop or 0.0
89
-
90
- if (last_accesses < 5) and (pop < 10.0):
91
- decision['error_reason'] = 'did not popular enough'
92
- return decision
93
-
94
- free_rses = self._rses
95
- available_reps = []
96
- reps = list_dataset_replicas(did[0], did[1])
97
- num_reps = 0
98
- for rep in reps:
99
- rse_attr = list_rse_attributes(rep['rse_id'])
100
- if 'type' not in rse_attr:
101
- continue
102
- if rse_attr['type'] != 'DATADISK':
103
- continue
104
- if rep['state'] == ReplicaState.AVAILABLE:
105
- if rep['rse_id'] in free_rses:
106
- free_rses.remove(rep['rse_id'])
107
- available_reps.append(rep['rse_id'])
108
- num_reps += 1
109
-
110
- decision['replica_rses'] = available_reps
111
- decision['num_replicas'] = num_reps
112
- if num_reps >= 5:
113
- decision['error_reason'] = 'more than 4 replicas already exist'
114
- return decision
115
-
116
- rse_ratios = {}
117
- space_info = self._fsc.get_rse_space()
118
- for rse_id in free_rses:
119
- rse_space = space_info[rse_id]
120
- penalty = self._penalties[rse_id]
121
- rse_ratios[rse_id] = float(rse_space['free']) / float(rse_space['total']) * 100.0 / penalty
122
-
123
- sorted_rses = sorted(rse_ratios.items(), key=itemgetter(1), reverse=True)
124
- decision['destination_rse'] = get_rse_name(rse_id=sorted_rses[0][0])
125
- decision['rse_ratios'] = sorted_rses
126
- self._penalties[sorted_rses[0][0]] = 10.0
127
-
128
- return decision
@@ -1,130 +0,0 @@
1
- # Copyright European Organization for Nuclear Research (CERN) since 2012
2
- #
3
- # Licensed under the Apache License, Version 2.0 (the "License");
4
- # you may not use this file except in compliance with the License.
5
- # You may obtain a copy of the License at
6
- #
7
- # http://www.apache.org/licenses/LICENSE-2.0
8
- #
9
- # Unless required by applicable law or agreed to in writing, software
10
- # distributed under the License is distributed on an "AS IS" BASIS,
11
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
- # See the License for the specific language governing permissions and
13
- # limitations under the License.
14
-
15
- import logging
16
- from operator import itemgetter
17
- from typing import TYPE_CHECKING, Any
18
-
19
- from rucio.common.config import config_get, config_get_int
20
- from rucio.common.exception import DataIdentifierNotFound
21
- from rucio.core.did import get_did
22
- from rucio.core.replica import list_dataset_replicas
23
- from rucio.core.rse import get_rse_name, list_rse_attributes
24
- from rucio.core.rse_expression_parser import parse_expression
25
- from rucio.daemons.c3po.collectors.free_space import FreeSpaceCollector
26
- from rucio.daemons.c3po.utils.expiring_dataset_cache import ExpiringDatasetCache
27
- from rucio.daemons.c3po.utils.popularity import get_popularity
28
- from rucio.db.sqla.constants import ReplicaState
29
-
30
- if TYPE_CHECKING:
31
- from rucio.common.types import InternalScope
32
-
33
-
34
- class PlacementAlgorithm:
35
- """
36
- Placement algorithm that focusses on free space on T2 DATADISK RSEs.
37
- """
38
- def __init__(self):
39
- self._fsc = FreeSpaceCollector()
40
- self._added_cache = ExpiringDatasetCache(config_get('c3po', 'redis_host'), config_get_int('c3po', 'redis_port'), timeout=86400)
41
-
42
- rse_expr = "tier=2&type=DATADISK"
43
- rse_attrs = parse_expression(rse_expr)
44
-
45
- self._rses = []
46
- for rse in rse_attrs:
47
- self._rses.append(rse['id'])
48
-
49
- self.__setup_penalties()
50
-
51
- def __setup_penalties(self) -> None:
52
- self._penalties = {}
53
- for rse_id in self._rses:
54
- self._penalties[rse_id] = 1.0
55
-
56
- def __update_penalties(self) -> None:
57
- for rse_id, penalty in self._penalties.items():
58
- if penalty > 1.0:
59
- self._penalties[rse_id] = penalty - 1
60
-
61
- def place(self, did: tuple['InternalScope', str]) -> dict[str, Any]:
62
- self.__update_penalties()
63
- decision: dict[str, Any] = {'did': '{}:{}'.format(did[0].internal, did[1])}
64
- if (self._added_cache.check_dataset(decision['did'])):
65
- decision['error_reason'] = 'already added replica for this did in the last 24h'
66
- return decision
67
-
68
- if (did[0].external is not None) and (not did[0].external.startswith('data')) and (not did[0].external.startswith('mc')):
69
- decision['error_reason'] = 'not a data or mc dataset'
70
- return decision
71
-
72
- try:
73
- meta = get_did(did[0], did[1])
74
- except DataIdentifierNotFound:
75
- decision['error_reason'] = 'did does not exist'
76
- return decision
77
- if meta['length'] is None:
78
- meta['length'] = 0
79
- if meta['bytes'] is None:
80
- meta['bytes'] = 0
81
- logging.debug('got %s:%s, num_files: %d, bytes: %d' % (did[0], did[1], meta['length'], meta['bytes']))
82
-
83
- decision['length'] = meta['length']
84
- decision['bytes'] = meta['bytes']
85
-
86
- pop = get_popularity(did)
87
- decision['popularity'] = pop or 0.0
88
-
89
- if (pop < 10.0):
90
- decision['error_reason'] = 'did not popular enough'
91
- return decision
92
-
93
- free_rses = self._rses
94
- available_reps = []
95
- reps = list_dataset_replicas(did[0], did[1])
96
- num_reps = 0
97
- for rep in reps:
98
- rse_attr = list_rse_attributes(rse_id=rep['rse_id'])
99
- if 'type' not in rse_attr:
100
- continue
101
- if rse_attr['type'] != 'DATADISK':
102
- continue
103
- if rep['state'] == ReplicaState.AVAILABLE:
104
- if rep['rse_id'] in free_rses:
105
- free_rses.remove(rep['rse_id'])
106
- available_reps.append(rep['rse_id'])
107
- num_reps += 1
108
-
109
- decision['replica_rses'] = available_reps
110
- decision['num_replicas'] = num_reps
111
-
112
- if num_reps >= 5:
113
- decision['error_reason'] = 'more than 4 replicas already exist'
114
- return decision
115
-
116
- rse_ratios = {}
117
- space_info = self._fsc.get_rse_space()
118
- for rse_id in free_rses:
119
- rse_space = space_info[rse_id]
120
- penalty = self._penalties[rse_id]
121
- rse_ratios[rse_id] = float(rse_space['free']) / float(rse_space['total']) * 100.0 / penalty
122
-
123
- sorted_rses = sorted(rse_ratios.items(), key=itemgetter(1), reverse=True)
124
- decision['destination_rse'] = get_rse_name(sorted_rses[0][0])
125
- decision['rse_ratios'] = sorted_rses
126
- self._penalties[sorted_rses[0][0]] = 10.0
127
-
128
- self._added_cache.add_dataset(decision['did'])
129
-
130
- return decision