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.
- rucio/alembicrevision.py +1 -1
- rucio/{daemons/c3po/collectors → cli}/__init__.py +1 -0
- rucio/cli/account.py +216 -0
- rucio-35.7.0.data/scripts/rucio → rucio/cli/bin_legacy/rucio.py +769 -486
- rucio-35.7.0.data/scripts/rucio-admin → rucio/cli/bin_legacy/rucio_admin.py +476 -423
- rucio/cli/command.py +272 -0
- rucio/cli/config.py +72 -0
- rucio/cli/did.py +191 -0
- rucio/cli/download.py +128 -0
- rucio/cli/lifetime_exception.py +33 -0
- rucio/cli/replica.py +162 -0
- rucio/cli/rse.py +293 -0
- rucio/cli/rule.py +158 -0
- rucio/cli/scope.py +40 -0
- rucio/cli/subscription.py +73 -0
- rucio/cli/upload.py +60 -0
- rucio/cli/utils.py +226 -0
- rucio/client/accountclient.py +0 -1
- rucio/client/baseclient.py +33 -24
- rucio/client/client.py +45 -1
- rucio/client/didclient.py +5 -3
- rucio/client/downloadclient.py +6 -8
- rucio/client/replicaclient.py +0 -2
- rucio/client/richclient.py +317 -0
- rucio/client/rseclient.py +4 -4
- rucio/client/uploadclient.py +26 -12
- rucio/common/bittorrent.py +234 -0
- rucio/common/cache.py +66 -29
- rucio/common/checksum.py +168 -0
- rucio/common/client.py +122 -0
- rucio/common/config.py +22 -35
- rucio/common/constants.py +61 -3
- rucio/common/didtype.py +72 -24
- rucio/common/dumper/__init__.py +45 -38
- rucio/common/dumper/consistency.py +75 -30
- rucio/common/dumper/data_models.py +63 -19
- rucio/common/dumper/path_parsing.py +19 -8
- rucio/common/exception.py +65 -8
- rucio/common/extra.py +5 -10
- rucio/common/logging.py +13 -13
- rucio/common/pcache.py +8 -7
- rucio/common/plugins.py +59 -27
- rucio/common/policy.py +12 -3
- rucio/common/schema/__init__.py +84 -34
- rucio/common/schema/generic.py +0 -17
- rucio/common/schema/generic_multi_vo.py +0 -17
- rucio/common/test_rucio_server.py +12 -6
- rucio/common/types.py +132 -52
- rucio/common/utils.py +93 -643
- rucio/core/account_limit.py +14 -12
- rucio/core/authentication.py +2 -2
- rucio/core/config.py +23 -42
- rucio/core/credential.py +14 -15
- rucio/core/did.py +5 -1
- rucio/core/did_meta_plugins/elasticsearch_meta.py +407 -0
- rucio/core/did_meta_plugins/filter_engine.py +62 -3
- rucio/core/did_meta_plugins/json_meta.py +2 -2
- rucio/core/did_meta_plugins/mongo_meta.py +43 -30
- rucio/core/did_meta_plugins/postgres_meta.py +75 -39
- rucio/core/identity.py +6 -5
- rucio/core/importer.py +4 -3
- rucio/core/lifetime_exception.py +2 -2
- rucio/core/lock.py +8 -7
- rucio/core/message.py +6 -0
- rucio/core/monitor.py +30 -29
- rucio/core/naming_convention.py +2 -2
- rucio/core/nongrid_trace.py +2 -2
- rucio/core/oidc.py +11 -9
- rucio/core/permission/__init__.py +79 -37
- rucio/core/permission/generic.py +1 -7
- rucio/core/permission/generic_multi_vo.py +1 -7
- rucio/core/quarantined_replica.py +4 -3
- rucio/core/replica.py +464 -139
- rucio/core/replica_sorter.py +55 -59
- rucio/core/request.py +34 -32
- rucio/core/rse.py +301 -97
- rucio/core/rse_counter.py +1 -2
- rucio/core/rse_expression_parser.py +7 -7
- rucio/core/rse_selector.py +9 -7
- rucio/core/rule.py +41 -40
- rucio/core/rule_grouping.py +42 -40
- rucio/core/scope.py +5 -4
- rucio/core/subscription.py +26 -28
- rucio/core/topology.py +11 -11
- rucio/core/trace.py +2 -2
- rucio/core/transfer.py +29 -15
- rucio/core/volatile_replica.py +4 -3
- rucio/daemons/atropos/atropos.py +1 -1
- rucio/daemons/auditor/__init__.py +2 -2
- rucio/daemons/auditor/srmdumps.py +6 -6
- rucio/daemons/automatix/automatix.py +32 -21
- rucio/daemons/badreplicas/necromancer.py +2 -2
- rucio/daemons/bb8/nuclei_background_rebalance.py +1 -1
- rucio/daemons/bb8/t2_background_rebalance.py +1 -1
- rucio/daemons/common.py +15 -25
- rucio/daemons/conveyor/finisher.py +2 -2
- rucio/daemons/conveyor/poller.py +18 -28
- rucio/daemons/conveyor/receiver.py +2 -2
- rucio/daemons/conveyor/stager.py +1 -0
- rucio/daemons/conveyor/submitter.py +3 -3
- rucio/daemons/hermes/hermes.py +91 -30
- rucio/daemons/judge/evaluator.py +2 -2
- rucio/daemons/oauthmanager/oauthmanager.py +3 -3
- rucio/daemons/reaper/dark_reaper.py +7 -3
- rucio/daemons/reaper/reaper.py +12 -16
- rucio/daemons/rsedecommissioner/config.py +1 -1
- rucio/daemons/rsedecommissioner/profiles/generic.py +5 -4
- rucio/daemons/rsedecommissioner/profiles/types.py +7 -6
- rucio/daemons/rsedecommissioner/rse_decommissioner.py +1 -1
- rucio/daemons/storage/consistency/actions.py +8 -6
- rucio/daemons/tracer/kronos.py +4 -4
- rucio/db/sqla/constants.py +5 -0
- rucio/db/sqla/migrate_repo/versions/1677d4d803c8_split_rse_availability_into_multiple.py +4 -4
- rucio/db/sqla/migrate_repo/versions/30d5206e9cad_increase_oauthrequest_redirect_msg_.py +37 -0
- rucio/db/sqla/models.py +157 -154
- rucio/db/sqla/session.py +58 -27
- rucio/db/sqla/types.py +2 -2
- rucio/db/sqla/util.py +2 -2
- rucio/gateway/account.py +18 -12
- rucio/gateway/account_limit.py +137 -60
- rucio/gateway/authentication.py +18 -12
- rucio/gateway/config.py +30 -20
- rucio/gateway/credential.py +9 -10
- rucio/gateway/did.py +70 -53
- rucio/gateway/dirac.py +6 -4
- rucio/gateway/exporter.py +3 -2
- rucio/gateway/heartbeat.py +6 -4
- rucio/gateway/identity.py +36 -51
- rucio/gateway/importer.py +3 -2
- rucio/gateway/lifetime_exception.py +3 -2
- rucio/gateway/meta_conventions.py +17 -6
- rucio/gateway/permission.py +4 -1
- rucio/gateway/quarantined_replica.py +3 -2
- rucio/gateway/replica.py +31 -22
- rucio/gateway/request.py +27 -18
- rucio/gateway/rse.py +69 -37
- rucio/gateway/rule.py +46 -26
- rucio/gateway/scope.py +3 -2
- rucio/gateway/subscription.py +14 -11
- rucio/gateway/vo.py +12 -8
- rucio/rse/__init__.py +3 -3
- rucio/rse/protocols/bittorrent.py +11 -1
- rucio/rse/protocols/cache.py +0 -11
- rucio/rse/protocols/dummy.py +0 -11
- rucio/rse/protocols/gfal.py +14 -9
- rucio/rse/protocols/globus.py +1 -1
- rucio/rse/protocols/http_cache.py +1 -1
- rucio/rse/protocols/posix.py +2 -2
- rucio/rse/protocols/protocol.py +84 -317
- rucio/rse/protocols/rclone.py +2 -1
- rucio/rse/protocols/rfio.py +10 -1
- rucio/rse/protocols/ssh.py +2 -1
- rucio/rse/protocols/storm.py +2 -13
- rucio/rse/protocols/webdav.py +74 -30
- rucio/rse/protocols/xrootd.py +2 -1
- rucio/rse/rsemanager.py +170 -53
- rucio/rse/translation.py +260 -0
- rucio/tests/common.py +23 -13
- rucio/tests/common_server.py +26 -9
- rucio/transfertool/bittorrent.py +15 -14
- rucio/transfertool/bittorrent_driver.py +5 -7
- rucio/transfertool/bittorrent_driver_qbittorrent.py +9 -8
- rucio/transfertool/fts3.py +20 -16
- rucio/transfertool/mock.py +2 -3
- rucio/vcsversion.py +4 -4
- rucio/version.py +7 -0
- rucio/web/rest/flaskapi/v1/accounts.py +17 -3
- rucio/web/rest/flaskapi/v1/auth.py +5 -5
- rucio/web/rest/flaskapi/v1/credentials.py +3 -2
- rucio/web/rest/flaskapi/v1/dids.py +21 -15
- rucio/web/rest/flaskapi/v1/identities.py +33 -9
- rucio/web/rest/flaskapi/v1/redirect.py +5 -4
- rucio/web/rest/flaskapi/v1/replicas.py +12 -8
- rucio/web/rest/flaskapi/v1/rses.py +15 -4
- rucio/web/rest/flaskapi/v1/traces.py +56 -19
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/alembic.ini.template +1 -1
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/alembic_offline.ini.template +1 -1
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/rucio.cfg.atlas.client.template +3 -2
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/rucio.cfg.template +3 -19
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/rucio_multi_vo.cfg.template +1 -18
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/requirements.server.txt +97 -68
- rucio-37.0.0.data/scripts/rucio +133 -0
- rucio-37.0.0.data/scripts/rucio-admin +97 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-atropos +2 -2
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-auditor +2 -1
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-automatix +2 -2
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-cache-client +17 -10
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-receiver +1 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-kronos +1 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-minos +2 -2
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-minos-temporary-expiration +2 -2
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-necromancer +2 -2
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-reaper +6 -6
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-transmogrifier +2 -2
- rucio-37.0.0.dist-info/METADATA +92 -0
- {rucio-35.7.0.dist-info → rucio-37.0.0.dist-info}/RECORD +237 -243
- {rucio-35.7.0.dist-info → rucio-37.0.0.dist-info}/licenses/AUTHORS.rst +3 -0
- rucio/common/schema/atlas.py +0 -413
- rucio/common/schema/belleii.py +0 -408
- rucio/common/schema/domatpc.py +0 -401
- rucio/common/schema/escape.py +0 -426
- rucio/common/schema/icecube.py +0 -406
- rucio/core/permission/atlas.py +0 -1348
- rucio/core/permission/belleii.py +0 -1077
- rucio/core/permission/escape.py +0 -1078
- rucio/daemons/c3po/algorithms/__init__.py +0 -13
- rucio/daemons/c3po/algorithms/simple.py +0 -134
- rucio/daemons/c3po/algorithms/t2_free_space.py +0 -128
- rucio/daemons/c3po/algorithms/t2_free_space_only_pop.py +0 -130
- rucio/daemons/c3po/algorithms/t2_free_space_only_pop_with_network.py +0 -294
- rucio/daemons/c3po/c3po.py +0 -371
- rucio/daemons/c3po/collectors/agis.py +0 -108
- rucio/daemons/c3po/collectors/free_space.py +0 -81
- rucio/daemons/c3po/collectors/jedi_did.py +0 -57
- rucio/daemons/c3po/collectors/mock_did.py +0 -51
- rucio/daemons/c3po/collectors/network_metrics.py +0 -71
- rucio/daemons/c3po/collectors/workload.py +0 -112
- rucio/daemons/c3po/utils/__init__.py +0 -13
- rucio/daemons/c3po/utils/dataset_cache.py +0 -50
- rucio/daemons/c3po/utils/expiring_dataset_cache.py +0 -56
- rucio/daemons/c3po/utils/expiring_list.py +0 -62
- rucio/daemons/c3po/utils/popularity.py +0 -85
- rucio/daemons/c3po/utils/timeseries.py +0 -89
- rucio/rse/protocols/gsiftp.py +0 -92
- rucio-35.7.0.data/scripts/rucio-c3po +0 -85
- rucio-35.7.0.dist-info/METADATA +0 -72
- /rucio/{daemons/c3po → cli/bin_legacy}/__init__.py +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/globus-config.yml.template +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/ldap.cfg.template +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_approval_request.tmpl +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_approved_admin.tmpl +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_approved_user.tmpl +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_denied_admin.tmpl +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_denied_user.tmpl +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/mail_templates/rule_ok_notification.tmpl +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/etc/rse-accounts.cfg.template +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/tools/bootstrap.py +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/tools/merge_rucio_configs.py +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/data/rucio/tools/reset_database.py +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-abacus-account +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-abacus-collection-replica +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-abacus-rse +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-bb8 +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-cache-consumer +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-finisher +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-poller +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-preparer +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-stager +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-submitter +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-conveyor-throttler +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-dark-reaper +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-dumper +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-follower +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-hermes +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-judge-cleaner +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-judge-evaluator +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-judge-injector +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-judge-repairer +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-oauth-manager +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-replica-recoverer +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-rse-decommissioner +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-storage-consistency-actions +0 -0
- {rucio-35.7.0.data → rucio-37.0.0.data}/scripts/rucio-undertaker +0 -0
- {rucio-35.7.0.dist-info → rucio-37.0.0.dist-info}/WHEEL +0 -0
- {rucio-35.7.0.dist-info → rucio-37.0.0.dist-info}/licenses/LICENSE +0 -0
- {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
|