rucio 35.7.0__py3-none-any.whl → 37.0.0__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 (266) 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/test_rucio_server.py +12 -6
  48. rucio/common/types.py +132 -52
  49. rucio/common/utils.py +93 -643
  50. rucio/core/account_limit.py +14 -12
  51. rucio/core/authentication.py +2 -2
  52. rucio/core/config.py +23 -42
  53. rucio/core/credential.py +14 -15
  54. rucio/core/did.py +5 -1
  55. rucio/core/did_meta_plugins/elasticsearch_meta.py +407 -0
  56. rucio/core/did_meta_plugins/filter_engine.py +62 -3
  57. rucio/core/did_meta_plugins/json_meta.py +2 -2
  58. rucio/core/did_meta_plugins/mongo_meta.py +43 -30
  59. rucio/core/did_meta_plugins/postgres_meta.py +75 -39
  60. rucio/core/identity.py +6 -5
  61. rucio/core/importer.py +4 -3
  62. rucio/core/lifetime_exception.py +2 -2
  63. rucio/core/lock.py +8 -7
  64. rucio/core/message.py +6 -0
  65. rucio/core/monitor.py +30 -29
  66. rucio/core/naming_convention.py +2 -2
  67. rucio/core/nongrid_trace.py +2 -2
  68. rucio/core/oidc.py +11 -9
  69. rucio/core/permission/__init__.py +79 -37
  70. rucio/core/permission/generic.py +1 -7
  71. rucio/core/permission/generic_multi_vo.py +1 -7
  72. rucio/core/quarantined_replica.py +4 -3
  73. rucio/core/replica.py +464 -139
  74. rucio/core/replica_sorter.py +55 -59
  75. rucio/core/request.py +34 -32
  76. rucio/core/rse.py +301 -97
  77. rucio/core/rse_counter.py +1 -2
  78. rucio/core/rse_expression_parser.py +7 -7
  79. rucio/core/rse_selector.py +9 -7
  80. rucio/core/rule.py +41 -40
  81. rucio/core/rule_grouping.py +42 -40
  82. rucio/core/scope.py +5 -4
  83. rucio/core/subscription.py +26 -28
  84. rucio/core/topology.py +11 -11
  85. rucio/core/trace.py +2 -2
  86. rucio/core/transfer.py +29 -15
  87. rucio/core/volatile_replica.py +4 -3
  88. rucio/daemons/atropos/atropos.py +1 -1
  89. rucio/daemons/auditor/__init__.py +2 -2
  90. rucio/daemons/auditor/srmdumps.py +6 -6
  91. rucio/daemons/automatix/automatix.py +32 -21
  92. rucio/daemons/badreplicas/necromancer.py +2 -2
  93. rucio/daemons/bb8/nuclei_background_rebalance.py +1 -1
  94. rucio/daemons/bb8/t2_background_rebalance.py +1 -1
  95. rucio/daemons/common.py +15 -25
  96. rucio/daemons/conveyor/finisher.py +2 -2
  97. rucio/daemons/conveyor/poller.py +18 -28
  98. rucio/daemons/conveyor/receiver.py +2 -2
  99. rucio/daemons/conveyor/stager.py +1 -0
  100. rucio/daemons/conveyor/submitter.py +3 -3
  101. rucio/daemons/hermes/hermes.py +91 -30
  102. rucio/daemons/judge/evaluator.py +2 -2
  103. rucio/daemons/oauthmanager/oauthmanager.py +3 -3
  104. rucio/daemons/reaper/dark_reaper.py +7 -3
  105. rucio/daemons/reaper/reaper.py +12 -16
  106. rucio/daemons/rsedecommissioner/config.py +1 -1
  107. rucio/daemons/rsedecommissioner/profiles/generic.py +5 -4
  108. rucio/daemons/rsedecommissioner/profiles/types.py +7 -6
  109. rucio/daemons/rsedecommissioner/rse_decommissioner.py +1 -1
  110. rucio/daemons/storage/consistency/actions.py +8 -6
  111. rucio/daemons/tracer/kronos.py +4 -4
  112. rucio/db/sqla/constants.py +5 -0
  113. rucio/db/sqla/migrate_repo/versions/1677d4d803c8_split_rse_availability_into_multiple.py +4 -4
  114. rucio/db/sqla/migrate_repo/versions/30d5206e9cad_increase_oauthrequest_redirect_msg_.py +37 -0
  115. rucio/db/sqla/models.py +157 -154
  116. rucio/db/sqla/session.py +58 -27
  117. rucio/db/sqla/types.py +2 -2
  118. rucio/db/sqla/util.py +2 -2
  119. rucio/gateway/account.py +18 -12
  120. rucio/gateway/account_limit.py +137 -60
  121. rucio/gateway/authentication.py +18 -12
  122. rucio/gateway/config.py +30 -20
  123. rucio/gateway/credential.py +9 -10
  124. rucio/gateway/did.py +70 -53
  125. rucio/gateway/dirac.py +6 -4
  126. rucio/gateway/exporter.py +3 -2
  127. rucio/gateway/heartbeat.py +6 -4
  128. rucio/gateway/identity.py +36 -51
  129. rucio/gateway/importer.py +3 -2
  130. rucio/gateway/lifetime_exception.py +3 -2
  131. rucio/gateway/meta_conventions.py +17 -6
  132. rucio/gateway/permission.py +4 -1
  133. rucio/gateway/quarantined_replica.py +3 -2
  134. rucio/gateway/replica.py +31 -22
  135. rucio/gateway/request.py +27 -18
  136. rucio/gateway/rse.py +69 -37
  137. rucio/gateway/rule.py +46 -26
  138. rucio/gateway/scope.py +3 -2
  139. rucio/gateway/subscription.py +14 -11
  140. rucio/gateway/vo.py +12 -8
  141. rucio/rse/__init__.py +3 -3
  142. rucio/rse/protocols/bittorrent.py +11 -1
  143. rucio/rse/protocols/cache.py +0 -11
  144. rucio/rse/protocols/dummy.py +0 -11
  145. rucio/rse/protocols/gfal.py +14 -9
  146. rucio/rse/protocols/globus.py +1 -1
  147. rucio/rse/protocols/http_cache.py +1 -1
  148. rucio/rse/protocols/posix.py +2 -2
  149. rucio/rse/protocols/protocol.py +84 -317
  150. rucio/rse/protocols/rclone.py +2 -1
  151. rucio/rse/protocols/rfio.py +10 -1
  152. rucio/rse/protocols/ssh.py +2 -1
  153. rucio/rse/protocols/storm.py +2 -13
  154. rucio/rse/protocols/webdav.py +74 -30
  155. rucio/rse/protocols/xrootd.py +2 -1
  156. rucio/rse/rsemanager.py +170 -53
  157. rucio/rse/translation.py +260 -0
  158. rucio/tests/common.py +23 -13
  159. rucio/tests/common_server.py +26 -9
  160. rucio/transfertool/bittorrent.py +15 -14
  161. rucio/transfertool/bittorrent_driver.py +5 -7
  162. rucio/transfertool/bittorrent_driver_qbittorrent.py +9 -8
  163. rucio/transfertool/fts3.py +20 -16
  164. rucio/transfertool/mock.py +2 -3
  165. rucio/vcsversion.py +4 -4
  166. rucio/version.py +7 -0
  167. rucio/web/rest/flaskapi/v1/accounts.py +17 -3
  168. rucio/web/rest/flaskapi/v1/auth.py +5 -5
  169. rucio/web/rest/flaskapi/v1/credentials.py +3 -2
  170. rucio/web/rest/flaskapi/v1/dids.py +21 -15
  171. rucio/web/rest/flaskapi/v1/identities.py +33 -9
  172. rucio/web/rest/flaskapi/v1/redirect.py +5 -4
  173. rucio/web/rest/flaskapi/v1/replicas.py +12 -8
  174. rucio/web/rest/flaskapi/v1/rses.py +15 -4
  175. rucio/web/rest/flaskapi/v1/traces.py +56 -19
  176. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/alembic.ini.template +1 -1
  177. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/alembic_offline.ini.template +1 -1
  178. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/rucio.cfg.atlas.client.template +3 -2
  179. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/rucio.cfg.template +3 -19
  180. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/rucio_multi_vo.cfg.template +1 -18
  181. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/requirements.server.txt +97 -68
  182. rucio-37.0.0.data/scripts/rucio +133 -0
  183. rucio-37.0.0.data/scripts/rucio-admin +97 -0
  184. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-atropos +2 -2
  185. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-auditor +2 -1
  186. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-automatix +2 -2
  187. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-cache-client +17 -10
  188. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-receiver +1 -0
  189. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-kronos +1 -0
  190. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-minos +2 -2
  191. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-minos-temporary-expiration +2 -2
  192. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-necromancer +2 -2
  193. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-reaper +6 -6
  194. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-transmogrifier +2 -2
  195. rucio-37.0.0.dist-info/METADATA +92 -0
  196. {rucio-35.7.0.dist-info → rucio-37.0.0.dist-info}/RECORD +237 -243
  197. {rucio-35.7.0.dist-info → rucio-37.0.0.dist-info}/licenses/AUTHORS.rst +3 -0
  198. rucio/common/schema/atlas.py +0 -413
  199. rucio/common/schema/belleii.py +0 -408
  200. rucio/common/schema/domatpc.py +0 -401
  201. rucio/common/schema/escape.py +0 -426
  202. rucio/common/schema/icecube.py +0 -406
  203. rucio/core/permission/atlas.py +0 -1348
  204. rucio/core/permission/belleii.py +0 -1077
  205. rucio/core/permission/escape.py +0 -1078
  206. rucio/daemons/c3po/algorithms/__init__.py +0 -13
  207. rucio/daemons/c3po/algorithms/simple.py +0 -134
  208. rucio/daemons/c3po/algorithms/t2_free_space.py +0 -128
  209. rucio/daemons/c3po/algorithms/t2_free_space_only_pop.py +0 -130
  210. rucio/daemons/c3po/algorithms/t2_free_space_only_pop_with_network.py +0 -294
  211. rucio/daemons/c3po/c3po.py +0 -371
  212. rucio/daemons/c3po/collectors/agis.py +0 -108
  213. rucio/daemons/c3po/collectors/free_space.py +0 -81
  214. rucio/daemons/c3po/collectors/jedi_did.py +0 -57
  215. rucio/daemons/c3po/collectors/mock_did.py +0 -51
  216. rucio/daemons/c3po/collectors/network_metrics.py +0 -71
  217. rucio/daemons/c3po/collectors/workload.py +0 -112
  218. rucio/daemons/c3po/utils/__init__.py +0 -13
  219. rucio/daemons/c3po/utils/dataset_cache.py +0 -50
  220. rucio/daemons/c3po/utils/expiring_dataset_cache.py +0 -56
  221. rucio/daemons/c3po/utils/expiring_list.py +0 -62
  222. rucio/daemons/c3po/utils/popularity.py +0 -85
  223. rucio/daemons/c3po/utils/timeseries.py +0 -89
  224. rucio/rse/protocols/gsiftp.py +0 -92
  225. rucio-35.7.0.data/scripts/rucio-c3po +0 -85
  226. rucio-35.7.0.dist-info/METADATA +0 -72
  227. /rucio/{daemons/c3po → cli/bin_legacy}/__init__.py +0 -0
  228. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/globus-config.yml.template +0 -0
  229. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/ldap.cfg.template +0 -0
  230. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_approval_request.tmpl +0 -0
  231. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_approved_admin.tmpl +0 -0
  232. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_approved_user.tmpl +0 -0
  233. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_denied_admin.tmpl +0 -0
  234. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_denied_user.tmpl +0 -0
  235. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_ok_notification.tmpl +0 -0
  236. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/rse-accounts.cfg.template +0 -0
  237. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/tools/bootstrap.py +0 -0
  238. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/tools/merge_rucio_configs.py +0 -0
  239. {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/tools/reset_database.py +0 -0
  240. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-abacus-account +0 -0
  241. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-abacus-collection-replica +0 -0
  242. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-abacus-rse +0 -0
  243. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-bb8 +0 -0
  244. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-cache-consumer +0 -0
  245. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-finisher +0 -0
  246. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-poller +0 -0
  247. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-preparer +0 -0
  248. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-stager +0 -0
  249. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-submitter +0 -0
  250. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-throttler +0 -0
  251. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-dark-reaper +0 -0
  252. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-dumper +0 -0
  253. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-follower +0 -0
  254. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-hermes +0 -0
  255. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-judge-cleaner +0 -0
  256. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-judge-evaluator +0 -0
  257. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-judge-injector +0 -0
  258. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-judge-repairer +0 -0
  259. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-oauth-manager +0 -0
  260. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-replica-recoverer +0 -0
  261. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-rse-decommissioner +0 -0
  262. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-storage-consistency-actions +0 -0
  263. {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-undertaker +0 -0
  264. {rucio-35.7.0.dist-info → rucio-37.0.0.dist-info}/WHEEL +0 -0
  265. {rucio-35.7.0.dist-info → rucio-37.0.0.dist-info}/licenses/LICENSE +0 -0
  266. {rucio-35.7.0.dist-info → rucio-37.0.0.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