rucio 37.0.0rc1__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/__init__.py +17 -0
- rucio/alembicrevision.py +15 -0
- rucio/cli/__init__.py +14 -0
- rucio/cli/account.py +216 -0
- rucio/cli/bin_legacy/__init__.py +13 -0
- rucio/cli/bin_legacy/rucio.py +2825 -0
- rucio/cli/bin_legacy/rucio_admin.py +2500 -0
- 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/__init__.py +15 -0
- rucio/client/accountclient.py +432 -0
- rucio/client/accountlimitclient.py +183 -0
- rucio/client/baseclient.py +983 -0
- rucio/client/client.py +120 -0
- rucio/client/configclient.py +126 -0
- rucio/client/credentialclient.py +59 -0
- rucio/client/didclient.py +868 -0
- rucio/client/diracclient.py +56 -0
- rucio/client/downloadclient.py +1783 -0
- rucio/client/exportclient.py +44 -0
- rucio/client/fileclient.py +50 -0
- rucio/client/importclient.py +42 -0
- rucio/client/lifetimeclient.py +90 -0
- rucio/client/lockclient.py +109 -0
- rucio/client/metaconventionsclient.py +140 -0
- rucio/client/pingclient.py +44 -0
- rucio/client/replicaclient.py +452 -0
- rucio/client/requestclient.py +125 -0
- rucio/client/richclient.py +317 -0
- rucio/client/rseclient.py +746 -0
- rucio/client/ruleclient.py +294 -0
- rucio/client/scopeclient.py +90 -0
- rucio/client/subscriptionclient.py +173 -0
- rucio/client/touchclient.py +82 -0
- rucio/client/uploadclient.py +969 -0
- rucio/common/__init__.py +13 -0
- rucio/common/bittorrent.py +234 -0
- rucio/common/cache.py +111 -0
- rucio/common/checksum.py +168 -0
- rucio/common/client.py +122 -0
- rucio/common/config.py +788 -0
- rucio/common/constants.py +217 -0
- rucio/common/constraints.py +17 -0
- rucio/common/didtype.py +237 -0
- rucio/common/dumper/__init__.py +342 -0
- rucio/common/dumper/consistency.py +497 -0
- rucio/common/dumper/data_models.py +362 -0
- rucio/common/dumper/path_parsing.py +75 -0
- rucio/common/exception.py +1208 -0
- rucio/common/extra.py +31 -0
- rucio/common/logging.py +420 -0
- rucio/common/pcache.py +1409 -0
- rucio/common/plugins.py +185 -0
- rucio/common/policy.py +93 -0
- rucio/common/schema/__init__.py +200 -0
- rucio/common/schema/generic.py +416 -0
- rucio/common/schema/generic_multi_vo.py +395 -0
- rucio/common/stomp_utils.py +423 -0
- rucio/common/stopwatch.py +55 -0
- rucio/common/test_rucio_server.py +154 -0
- rucio/common/types.py +483 -0
- rucio/common/utils.py +1688 -0
- rucio/core/__init__.py +13 -0
- rucio/core/account.py +496 -0
- rucio/core/account_counter.py +236 -0
- rucio/core/account_limit.py +425 -0
- rucio/core/authentication.py +620 -0
- rucio/core/config.py +437 -0
- rucio/core/credential.py +224 -0
- rucio/core/did.py +3004 -0
- rucio/core/did_meta_plugins/__init__.py +252 -0
- rucio/core/did_meta_plugins/did_column_meta.py +331 -0
- rucio/core/did_meta_plugins/did_meta_plugin_interface.py +165 -0
- rucio/core/did_meta_plugins/elasticsearch_meta.py +407 -0
- rucio/core/did_meta_plugins/filter_engine.py +672 -0
- rucio/core/did_meta_plugins/json_meta.py +240 -0
- rucio/core/did_meta_plugins/mongo_meta.py +229 -0
- rucio/core/did_meta_plugins/postgres_meta.py +352 -0
- rucio/core/dirac.py +237 -0
- rucio/core/distance.py +187 -0
- rucio/core/exporter.py +59 -0
- rucio/core/heartbeat.py +363 -0
- rucio/core/identity.py +301 -0
- rucio/core/importer.py +260 -0
- rucio/core/lifetime_exception.py +377 -0
- rucio/core/lock.py +577 -0
- rucio/core/message.py +288 -0
- rucio/core/meta_conventions.py +203 -0
- rucio/core/monitor.py +448 -0
- rucio/core/naming_convention.py +195 -0
- rucio/core/nongrid_trace.py +136 -0
- rucio/core/oidc.py +1463 -0
- rucio/core/permission/__init__.py +161 -0
- rucio/core/permission/generic.py +1124 -0
- rucio/core/permission/generic_multi_vo.py +1144 -0
- rucio/core/quarantined_replica.py +224 -0
- rucio/core/replica.py +4483 -0
- rucio/core/replica_sorter.py +362 -0
- rucio/core/request.py +3091 -0
- rucio/core/rse.py +2079 -0
- rucio/core/rse_counter.py +185 -0
- rucio/core/rse_expression_parser.py +459 -0
- rucio/core/rse_selector.py +304 -0
- rucio/core/rule.py +4484 -0
- rucio/core/rule_grouping.py +1620 -0
- rucio/core/scope.py +181 -0
- rucio/core/subscription.py +362 -0
- rucio/core/topology.py +490 -0
- rucio/core/trace.py +375 -0
- rucio/core/transfer.py +1531 -0
- rucio/core/vo.py +169 -0
- rucio/core/volatile_replica.py +151 -0
- rucio/daemons/__init__.py +13 -0
- rucio/daemons/abacus/__init__.py +13 -0
- rucio/daemons/abacus/account.py +116 -0
- rucio/daemons/abacus/collection_replica.py +124 -0
- rucio/daemons/abacus/rse.py +117 -0
- rucio/daemons/atropos/__init__.py +13 -0
- rucio/daemons/atropos/atropos.py +242 -0
- rucio/daemons/auditor/__init__.py +289 -0
- rucio/daemons/auditor/hdfs.py +97 -0
- rucio/daemons/auditor/srmdumps.py +355 -0
- rucio/daemons/automatix/__init__.py +13 -0
- rucio/daemons/automatix/automatix.py +304 -0
- rucio/daemons/badreplicas/__init__.py +13 -0
- rucio/daemons/badreplicas/minos.py +322 -0
- rucio/daemons/badreplicas/minos_temporary_expiration.py +171 -0
- rucio/daemons/badreplicas/necromancer.py +196 -0
- rucio/daemons/bb8/__init__.py +13 -0
- rucio/daemons/bb8/bb8.py +353 -0
- rucio/daemons/bb8/common.py +759 -0
- rucio/daemons/bb8/nuclei_background_rebalance.py +153 -0
- rucio/daemons/bb8/t2_background_rebalance.py +153 -0
- rucio/daemons/cache/__init__.py +13 -0
- rucio/daemons/cache/consumer.py +133 -0
- rucio/daemons/common.py +405 -0
- rucio/daemons/conveyor/__init__.py +13 -0
- rucio/daemons/conveyor/common.py +562 -0
- rucio/daemons/conveyor/finisher.py +529 -0
- rucio/daemons/conveyor/poller.py +394 -0
- rucio/daemons/conveyor/preparer.py +205 -0
- rucio/daemons/conveyor/receiver.py +179 -0
- rucio/daemons/conveyor/stager.py +133 -0
- rucio/daemons/conveyor/submitter.py +403 -0
- rucio/daemons/conveyor/throttler.py +532 -0
- rucio/daemons/follower/__init__.py +13 -0
- rucio/daemons/follower/follower.py +101 -0
- rucio/daemons/hermes/__init__.py +13 -0
- rucio/daemons/hermes/hermes.py +534 -0
- rucio/daemons/judge/__init__.py +13 -0
- rucio/daemons/judge/cleaner.py +159 -0
- rucio/daemons/judge/evaluator.py +185 -0
- rucio/daemons/judge/injector.py +162 -0
- rucio/daemons/judge/repairer.py +154 -0
- rucio/daemons/oauthmanager/__init__.py +13 -0
- rucio/daemons/oauthmanager/oauthmanager.py +198 -0
- rucio/daemons/reaper/__init__.py +13 -0
- rucio/daemons/reaper/dark_reaper.py +282 -0
- rucio/daemons/reaper/reaper.py +739 -0
- rucio/daemons/replicarecoverer/__init__.py +13 -0
- rucio/daemons/replicarecoverer/suspicious_replica_recoverer.py +626 -0
- rucio/daemons/rsedecommissioner/__init__.py +13 -0
- rucio/daemons/rsedecommissioner/config.py +81 -0
- rucio/daemons/rsedecommissioner/profiles/__init__.py +24 -0
- rucio/daemons/rsedecommissioner/profiles/atlas.py +60 -0
- rucio/daemons/rsedecommissioner/profiles/generic.py +452 -0
- rucio/daemons/rsedecommissioner/profiles/types.py +93 -0
- rucio/daemons/rsedecommissioner/rse_decommissioner.py +280 -0
- rucio/daemons/storage/__init__.py +13 -0
- rucio/daemons/storage/consistency/__init__.py +13 -0
- rucio/daemons/storage/consistency/actions.py +848 -0
- rucio/daemons/tracer/__init__.py +13 -0
- rucio/daemons/tracer/kronos.py +511 -0
- rucio/daemons/transmogrifier/__init__.py +13 -0
- rucio/daemons/transmogrifier/transmogrifier.py +762 -0
- rucio/daemons/undertaker/__init__.py +13 -0
- rucio/daemons/undertaker/undertaker.py +137 -0
- rucio/db/__init__.py +13 -0
- rucio/db/sqla/__init__.py +52 -0
- rucio/db/sqla/constants.py +206 -0
- rucio/db/sqla/migrate_repo/__init__.py +13 -0
- rucio/db/sqla/migrate_repo/env.py +110 -0
- rucio/db/sqla/migrate_repo/versions/01eaf73ab656_add_new_rule_notification_state_progress.py +70 -0
- rucio/db/sqla/migrate_repo/versions/0437a40dbfd1_add_eol_at_in_rules.py +47 -0
- rucio/db/sqla/migrate_repo/versions/0f1adb7a599a_create_transfer_hops_table.py +59 -0
- rucio/db/sqla/migrate_repo/versions/102efcf145f4_added_stuck_at_column_to_rules.py +43 -0
- rucio/db/sqla/migrate_repo/versions/13d4f70c66a9_introduce_transfer_limits.py +91 -0
- rucio/db/sqla/migrate_repo/versions/140fef722e91_cleanup_distances_table.py +76 -0
- rucio/db/sqla/migrate_repo/versions/14ec5aeb64cf_add_request_external_host.py +43 -0
- rucio/db/sqla/migrate_repo/versions/156fb5b5a14_add_request_type_to_requests_idx.py +50 -0
- rucio/db/sqla/migrate_repo/versions/1677d4d803c8_split_rse_availability_into_multiple.py +68 -0
- rucio/db/sqla/migrate_repo/versions/16a0aca82e12_create_index_on_table_replicas_path.py +40 -0
- rucio/db/sqla/migrate_repo/versions/1803333ac20f_adding_provenance_and_phys_group.py +45 -0
- rucio/db/sqla/migrate_repo/versions/1a29d6a9504c_add_didtype_chck_to_requests.py +60 -0
- rucio/db/sqla/migrate_repo/versions/1a80adff031a_create_index_on_rules_hist_recent.py +40 -0
- rucio/db/sqla/migrate_repo/versions/1c45d9730ca6_increase_identity_length.py +140 -0
- rucio/db/sqla/migrate_repo/versions/1d1215494e95_add_quarantined_replicas_table.py +73 -0
- rucio/db/sqla/migrate_repo/versions/1d96f484df21_asynchronous_rules_and_rule_approval.py +74 -0
- rucio/db/sqla/migrate_repo/versions/1f46c5f240ac_add_bytes_column_to_bad_replicas.py +43 -0
- rucio/db/sqla/migrate_repo/versions/1fc15ab60d43_add_message_history_table.py +50 -0
- rucio/db/sqla/migrate_repo/versions/2190e703eb6e_move_rse_settings_to_rse_attributes.py +134 -0
- rucio/db/sqla/migrate_repo/versions/21d6b9dc9961_add_mismatch_scheme_state_to_requests.py +64 -0
- rucio/db/sqla/migrate_repo/versions/22cf51430c78_add_availability_column_to_table_rses.py +39 -0
- rucio/db/sqla/migrate_repo/versions/22d887e4ec0a_create_sources_table.py +64 -0
- rucio/db/sqla/migrate_repo/versions/25821a8a45a3_remove_unique_constraint_on_requests.py +51 -0
- rucio/db/sqla/migrate_repo/versions/25fc855625cf_added_unique_constraint_to_rules.py +41 -0
- rucio/db/sqla/migrate_repo/versions/269fee20dee9_add_repair_cnt_to_locks.py +43 -0
- rucio/db/sqla/migrate_repo/versions/271a46ea6244_add_ignore_availability_column_to_rules.py +44 -0
- rucio/db/sqla/migrate_repo/versions/277b5fbb41d3_switch_heartbeats_executable.py +53 -0
- rucio/db/sqla/migrate_repo/versions/27e3a68927fb_remove_replicas_tombstone_and_replicas_.py +38 -0
- rucio/db/sqla/migrate_repo/versions/2854cd9e168_added_rule_id_column.py +47 -0
- rucio/db/sqla/migrate_repo/versions/295289b5a800_processed_by_and__at_in_requests.py +45 -0
- rucio/db/sqla/migrate_repo/versions/2962ece31cf4_add_nbaccesses_column_in_the_did_table.py +45 -0
- rucio/db/sqla/migrate_repo/versions/2af3291ec4c_added_replicas_history_table.py +57 -0
- rucio/db/sqla/migrate_repo/versions/2b69addda658_add_columns_for_third_party_copy_read_.py +45 -0
- rucio/db/sqla/migrate_repo/versions/2b8e7bcb4783_add_config_table.py +69 -0
- rucio/db/sqla/migrate_repo/versions/2ba5229cb54c_add_submitted_at_to_requests_table.py +43 -0
- rucio/db/sqla/migrate_repo/versions/2cbee484dcf9_added_column_volume_to_rse_transfer_.py +42 -0
- rucio/db/sqla/migrate_repo/versions/2edee4a83846_add_source_to_requests_and_requests_.py +47 -0
- rucio/db/sqla/migrate_repo/versions/2eef46be23d4_change_tokens_pk.py +46 -0
- rucio/db/sqla/migrate_repo/versions/2f648fc909f3_index_in_rule_history_on_scope_name.py +40 -0
- rucio/db/sqla/migrate_repo/versions/3082b8cef557_add_naming_convention_table_and_closed_.py +67 -0
- rucio/db/sqla/migrate_repo/versions/30d5206e9cad_increase_oauthrequest_redirect_msg_.py +37 -0
- rucio/db/sqla/migrate_repo/versions/30fa38b6434e_add_index_on_service_column_in_the_message_table.py +44 -0
- rucio/db/sqla/migrate_repo/versions/3152492b110b_added_staging_area_column.py +77 -0
- rucio/db/sqla/migrate_repo/versions/32c7d2783f7e_create_bad_replicas_table.py +60 -0
- rucio/db/sqla/migrate_repo/versions/3345511706b8_replicas_table_pk_definition_is_in_.py +72 -0
- rucio/db/sqla/migrate_repo/versions/35ef10d1e11b_change_index_on_table_requests.py +42 -0
- rucio/db/sqla/migrate_repo/versions/379a19b5332d_create_rse_limits_table.py +65 -0
- rucio/db/sqla/migrate_repo/versions/384b96aa0f60_created_rule_history_tables.py +133 -0
- rucio/db/sqla/migrate_repo/versions/3ac1660a1a72_extend_distance_table.py +55 -0
- rucio/db/sqla/migrate_repo/versions/3ad36e2268b0_create_collection_replicas_updates_table.py +76 -0
- rucio/db/sqla/migrate_repo/versions/3c9df354071b_extend_waiting_request_state.py +60 -0
- rucio/db/sqla/migrate_repo/versions/3d9813fab443_add_a_new_state_lost_in_badfilesstatus.py +44 -0
- rucio/db/sqla/migrate_repo/versions/40ad39ce3160_add_transferred_at_to_requests_table.py +43 -0
- rucio/db/sqla/migrate_repo/versions/4207be2fd914_add_notification_column_to_rules.py +64 -0
- rucio/db/sqla/migrate_repo/versions/42db2617c364_create_index_on_requests_external_id.py +40 -0
- rucio/db/sqla/migrate_repo/versions/436827b13f82_added_column_activity_to_table_requests.py +43 -0
- rucio/db/sqla/migrate_repo/versions/44278720f774_update_requests_typ_sta_upd_idx_index.py +44 -0
- rucio/db/sqla/migrate_repo/versions/45378a1e76a8_create_collection_replica_table.py +78 -0
- rucio/db/sqla/migrate_repo/versions/469d262be19_removing_created_at_index.py +41 -0
- rucio/db/sqla/migrate_repo/versions/4783c1f49cb4_create_distance_table.py +59 -0
- rucio/db/sqla/migrate_repo/versions/49a21b4d4357_create_index_on_table_tokens.py +44 -0
- rucio/db/sqla/migrate_repo/versions/4a2cbedda8b9_add_source_replica_expression_column_to_.py +43 -0
- rucio/db/sqla/migrate_repo/versions/4a7182d9578b_added_bytes_length_accessed_at_columns.py +49 -0
- rucio/db/sqla/migrate_repo/versions/4bab9edd01fc_create_index_on_requests_rule_id.py +40 -0
- rucio/db/sqla/migrate_repo/versions/4c3a4acfe006_new_attr_account_table.py +63 -0
- rucio/db/sqla/migrate_repo/versions/4cf0a2e127d4_adding_transient_metadata.py +43 -0
- rucio/db/sqla/migrate_repo/versions/4df2c5ddabc0_remove_temporary_dids.py +55 -0
- rucio/db/sqla/migrate_repo/versions/50280c53117c_add_qos_class_to_rse.py +45 -0
- rucio/db/sqla/migrate_repo/versions/52153819589c_add_rse_id_to_replicas_table.py +43 -0
- rucio/db/sqla/migrate_repo/versions/52fd9f4916fa_added_activity_to_rules.py +43 -0
- rucio/db/sqla/migrate_repo/versions/53b479c3cb0f_fix_did_meta_table_missing_updated_at_.py +45 -0
- rucio/db/sqla/migrate_repo/versions/5673b4b6e843_add_wfms_metadata_to_rule_tables.py +47 -0
- rucio/db/sqla/migrate_repo/versions/575767d9f89_added_source_history_table.py +58 -0
- rucio/db/sqla/migrate_repo/versions/58bff7008037_add_started_at_to_requests.py +45 -0
- rucio/db/sqla/migrate_repo/versions/58c8b78301ab_rename_callback_to_message.py +106 -0
- rucio/db/sqla/migrate_repo/versions/5f139f77382a_added_child_rule_id_column.py +55 -0
- rucio/db/sqla/migrate_repo/versions/688ef1840840_adding_did_meta_table.py +50 -0
- rucio/db/sqla/migrate_repo/versions/6e572a9bfbf3_add_new_split_container_column_to_rules.py +47 -0
- rucio/db/sqla/migrate_repo/versions/70587619328_add_comment_column_for_subscriptions.py +43 -0
- rucio/db/sqla/migrate_repo/versions/739064d31565_remove_history_table_pks.py +41 -0
- rucio/db/sqla/migrate_repo/versions/7541902bf173_add_didsfollowed_and_followevents_table.py +91 -0
- rucio/db/sqla/migrate_repo/versions/7ec22226cdbf_new_replica_state_for_temporary_.py +72 -0
- rucio/db/sqla/migrate_repo/versions/810a41685bc1_added_columns_rse_transfer_limits.py +49 -0
- rucio/db/sqla/migrate_repo/versions/83f991c63a93_correct_rse_expression_length.py +43 -0
- rucio/db/sqla/migrate_repo/versions/8523998e2e76_increase_size_of_extended_attributes_.py +43 -0
- rucio/db/sqla/migrate_repo/versions/8ea9122275b1_adding_missing_function_based_indices.py +53 -0
- rucio/db/sqla/migrate_repo/versions/90f47792bb76_add_clob_payload_to_messages.py +45 -0
- rucio/db/sqla/migrate_repo/versions/914b8f02df38_new_table_for_lifetime_model_exceptions.py +68 -0
- rucio/db/sqla/migrate_repo/versions/94a5961ddbf2_add_estimator_columns.py +45 -0
- rucio/db/sqla/migrate_repo/versions/9a1b149a2044_add_saml_identity_type.py +94 -0
- rucio/db/sqla/migrate_repo/versions/9a45bc4ea66d_add_vp_table.py +54 -0
- rucio/db/sqla/migrate_repo/versions/9eb936a81eb1_true_is_true.py +72 -0
- rucio/db/sqla/migrate_repo/versions/a08fa8de1545_transfer_stats_table.py +55 -0
- rucio/db/sqla/migrate_repo/versions/a118956323f8_added_vo_table_and_vo_col_to_rse.py +76 -0
- rucio/db/sqla/migrate_repo/versions/a193a275255c_add_status_column_in_messages.py +47 -0
- rucio/db/sqla/migrate_repo/versions/a5f6f6e928a7_1_7_0.py +121 -0
- rucio/db/sqla/migrate_repo/versions/a616581ee47_added_columns_to_table_requests.py +59 -0
- rucio/db/sqla/migrate_repo/versions/a6eb23955c28_state_idx_non_functional.py +52 -0
- rucio/db/sqla/migrate_repo/versions/a74275a1ad30_added_global_quota_table.py +54 -0
- rucio/db/sqla/migrate_repo/versions/a93e4e47bda_heartbeats.py +64 -0
- rucio/db/sqla/migrate_repo/versions/ae2a56fcc89_added_comment_column_to_rules.py +49 -0
- rucio/db/sqla/migrate_repo/versions/b0070f3695c8_add_deletedidmeta_table.py +57 -0
- rucio/db/sqla/migrate_repo/versions/b4293a99f344_added_column_identity_to_table_tokens.py +43 -0
- rucio/db/sqla/migrate_repo/versions/b5493606bbf5_fix_primary_key_for_subscription_history.py +41 -0
- rucio/db/sqla/migrate_repo/versions/b7d287de34fd_removal_of_replicastate_source.py +91 -0
- rucio/db/sqla/migrate_repo/versions/b818052fa670_add_index_to_quarantined_replicas.py +40 -0
- rucio/db/sqla/migrate_repo/versions/b8caac94d7f0_add_comments_column_for_subscriptions_.py +43 -0
- rucio/db/sqla/migrate_repo/versions/b96a1c7e1cc4_new_bad_pfns_table_and_bad_replicas_.py +143 -0
- rucio/db/sqla/migrate_repo/versions/bb695f45c04_extend_request_state.py +76 -0
- rucio/db/sqla/migrate_repo/versions/bc68e9946deb_add_staging_timestamps_to_request.py +50 -0
- rucio/db/sqla/migrate_repo/versions/bf3baa1c1474_correct_pk_and_idx_for_history_tables.py +72 -0
- rucio/db/sqla/migrate_repo/versions/c0937668555f_add_qos_policy_map_table.py +55 -0
- rucio/db/sqla/migrate_repo/versions/c129ccdb2d5_add_lumiblocknr_to_dids.py +43 -0
- rucio/db/sqla/migrate_repo/versions/ccdbcd48206e_add_did_type_column_index_on_did_meta_.py +65 -0
- rucio/db/sqla/migrate_repo/versions/cebad904c4dd_new_payload_column_for_heartbeats.py +47 -0
- rucio/db/sqla/migrate_repo/versions/d1189a09c6e0_oauth2_0_and_jwt_feature_support_adding_.py +146 -0
- rucio/db/sqla/migrate_repo/versions/d23453595260_extend_request_state_for_preparer.py +104 -0
- rucio/db/sqla/migrate_repo/versions/d6dceb1de2d_added_purge_column_to_rules.py +44 -0
- rucio/db/sqla/migrate_repo/versions/d6e2c3b2cf26_remove_third_party_copy_column_from_rse.py +43 -0
- rucio/db/sqla/migrate_repo/versions/d91002c5841_new_account_limits_table.py +103 -0
- rucio/db/sqla/migrate_repo/versions/e138c364ebd0_extending_columns_for_filter_and_.py +49 -0
- rucio/db/sqla/migrate_repo/versions/e59300c8b179_support_for_archive.py +104 -0
- rucio/db/sqla/migrate_repo/versions/f1b14a8c2ac1_postgres_use_check_constraints.py +29 -0
- rucio/db/sqla/migrate_repo/versions/f41ffe206f37_oracle_global_temporary_tables.py +74 -0
- rucio/db/sqla/migrate_repo/versions/f85a2962b021_adding_transfertool_column_to_requests_.py +47 -0
- rucio/db/sqla/migrate_repo/versions/fa7a7d78b602_increase_refresh_token_size.py +43 -0
- rucio/db/sqla/migrate_repo/versions/fb28a95fe288_add_replicas_rse_id_tombstone_idx.py +37 -0
- rucio/db/sqla/migrate_repo/versions/fe1a65b176c9_set_third_party_copy_read_and_write_.py +43 -0
- rucio/db/sqla/migrate_repo/versions/fe8ea2fa9788_added_third_party_copy_column_to_rse_.py +43 -0
- rucio/db/sqla/models.py +1743 -0
- rucio/db/sqla/sautils.py +55 -0
- rucio/db/sqla/session.py +529 -0
- rucio/db/sqla/types.py +206 -0
- rucio/db/sqla/util.py +543 -0
- rucio/gateway/__init__.py +13 -0
- rucio/gateway/account.py +345 -0
- rucio/gateway/account_limit.py +363 -0
- rucio/gateway/authentication.py +381 -0
- rucio/gateway/config.py +227 -0
- rucio/gateway/credential.py +70 -0
- rucio/gateway/did.py +987 -0
- rucio/gateway/dirac.py +83 -0
- rucio/gateway/exporter.py +60 -0
- rucio/gateway/heartbeat.py +76 -0
- rucio/gateway/identity.py +189 -0
- rucio/gateway/importer.py +46 -0
- rucio/gateway/lifetime_exception.py +121 -0
- rucio/gateway/lock.py +153 -0
- rucio/gateway/meta_conventions.py +98 -0
- rucio/gateway/permission.py +74 -0
- rucio/gateway/quarantined_replica.py +79 -0
- rucio/gateway/replica.py +538 -0
- rucio/gateway/request.py +330 -0
- rucio/gateway/rse.py +632 -0
- rucio/gateway/rule.py +437 -0
- rucio/gateway/scope.py +100 -0
- rucio/gateway/subscription.py +280 -0
- rucio/gateway/vo.py +126 -0
- rucio/rse/__init__.py +96 -0
- rucio/rse/protocols/__init__.py +13 -0
- rucio/rse/protocols/bittorrent.py +194 -0
- rucio/rse/protocols/cache.py +111 -0
- rucio/rse/protocols/dummy.py +100 -0
- rucio/rse/protocols/gfal.py +708 -0
- rucio/rse/protocols/globus.py +243 -0
- rucio/rse/protocols/http_cache.py +82 -0
- rucio/rse/protocols/mock.py +123 -0
- rucio/rse/protocols/ngarc.py +209 -0
- rucio/rse/protocols/posix.py +250 -0
- rucio/rse/protocols/protocol.py +361 -0
- rucio/rse/protocols/rclone.py +365 -0
- rucio/rse/protocols/rfio.py +145 -0
- rucio/rse/protocols/srm.py +338 -0
- rucio/rse/protocols/ssh.py +414 -0
- rucio/rse/protocols/storm.py +195 -0
- rucio/rse/protocols/webdav.py +594 -0
- rucio/rse/protocols/xrootd.py +302 -0
- rucio/rse/rsemanager.py +881 -0
- rucio/rse/translation.py +260 -0
- rucio/tests/__init__.py +13 -0
- rucio/tests/common.py +280 -0
- rucio/tests/common_server.py +149 -0
- rucio/transfertool/__init__.py +13 -0
- rucio/transfertool/bittorrent.py +200 -0
- rucio/transfertool/bittorrent_driver.py +50 -0
- rucio/transfertool/bittorrent_driver_qbittorrent.py +134 -0
- rucio/transfertool/fts3.py +1600 -0
- rucio/transfertool/fts3_plugins.py +152 -0
- rucio/transfertool/globus.py +201 -0
- rucio/transfertool/globus_library.py +181 -0
- rucio/transfertool/mock.py +89 -0
- rucio/transfertool/transfertool.py +221 -0
- rucio/vcsversion.py +11 -0
- rucio/version.py +45 -0
- rucio/web/__init__.py +13 -0
- rucio/web/rest/__init__.py +13 -0
- rucio/web/rest/flaskapi/__init__.py +13 -0
- rucio/web/rest/flaskapi/authenticated_bp.py +27 -0
- rucio/web/rest/flaskapi/v1/__init__.py +13 -0
- rucio/web/rest/flaskapi/v1/accountlimits.py +236 -0
- rucio/web/rest/flaskapi/v1/accounts.py +1103 -0
- rucio/web/rest/flaskapi/v1/archives.py +102 -0
- rucio/web/rest/flaskapi/v1/auth.py +1644 -0
- rucio/web/rest/flaskapi/v1/common.py +426 -0
- rucio/web/rest/flaskapi/v1/config.py +304 -0
- rucio/web/rest/flaskapi/v1/credentials.py +213 -0
- rucio/web/rest/flaskapi/v1/dids.py +2340 -0
- rucio/web/rest/flaskapi/v1/dirac.py +116 -0
- rucio/web/rest/flaskapi/v1/export.py +75 -0
- rucio/web/rest/flaskapi/v1/heartbeats.py +127 -0
- rucio/web/rest/flaskapi/v1/identities.py +285 -0
- rucio/web/rest/flaskapi/v1/import.py +132 -0
- rucio/web/rest/flaskapi/v1/lifetime_exceptions.py +312 -0
- rucio/web/rest/flaskapi/v1/locks.py +358 -0
- rucio/web/rest/flaskapi/v1/main.py +91 -0
- rucio/web/rest/flaskapi/v1/meta_conventions.py +241 -0
- rucio/web/rest/flaskapi/v1/metrics.py +36 -0
- rucio/web/rest/flaskapi/v1/nongrid_traces.py +97 -0
- rucio/web/rest/flaskapi/v1/ping.py +88 -0
- rucio/web/rest/flaskapi/v1/redirect.py +366 -0
- rucio/web/rest/flaskapi/v1/replicas.py +1894 -0
- rucio/web/rest/flaskapi/v1/requests.py +998 -0
- rucio/web/rest/flaskapi/v1/rses.py +2250 -0
- rucio/web/rest/flaskapi/v1/rules.py +854 -0
- rucio/web/rest/flaskapi/v1/scopes.py +159 -0
- rucio/web/rest/flaskapi/v1/subscriptions.py +650 -0
- rucio/web/rest/flaskapi/v1/templates/auth_crash.html +80 -0
- rucio/web/rest/flaskapi/v1/templates/auth_granted.html +82 -0
- rucio/web/rest/flaskapi/v1/traces.py +137 -0
- rucio/web/rest/flaskapi/v1/types.py +20 -0
- rucio/web/rest/flaskapi/v1/vos.py +278 -0
- rucio/web/rest/main.py +18 -0
- rucio/web/rest/metrics.py +27 -0
- rucio/web/rest/ping.py +27 -0
- rucio-37.0.0rc1.data/data/rucio/etc/alembic.ini.template +71 -0
- rucio-37.0.0rc1.data/data/rucio/etc/alembic_offline.ini.template +74 -0
- rucio-37.0.0rc1.data/data/rucio/etc/globus-config.yml.template +5 -0
- rucio-37.0.0rc1.data/data/rucio/etc/ldap.cfg.template +30 -0
- rucio-37.0.0rc1.data/data/rucio/etc/mail_templates/rule_approval_request.tmpl +38 -0
- rucio-37.0.0rc1.data/data/rucio/etc/mail_templates/rule_approved_admin.tmpl +4 -0
- rucio-37.0.0rc1.data/data/rucio/etc/mail_templates/rule_approved_user.tmpl +17 -0
- rucio-37.0.0rc1.data/data/rucio/etc/mail_templates/rule_denied_admin.tmpl +6 -0
- rucio-37.0.0rc1.data/data/rucio/etc/mail_templates/rule_denied_user.tmpl +17 -0
- rucio-37.0.0rc1.data/data/rucio/etc/mail_templates/rule_ok_notification.tmpl +19 -0
- rucio-37.0.0rc1.data/data/rucio/etc/rse-accounts.cfg.template +25 -0
- rucio-37.0.0rc1.data/data/rucio/etc/rucio.cfg.atlas.client.template +43 -0
- rucio-37.0.0rc1.data/data/rucio/etc/rucio.cfg.template +241 -0
- rucio-37.0.0rc1.data/data/rucio/etc/rucio_multi_vo.cfg.template +217 -0
- rucio-37.0.0rc1.data/data/rucio/requirements.server.txt +297 -0
- rucio-37.0.0rc1.data/data/rucio/tools/bootstrap.py +34 -0
- rucio-37.0.0rc1.data/data/rucio/tools/merge_rucio_configs.py +144 -0
- rucio-37.0.0rc1.data/data/rucio/tools/reset_database.py +40 -0
- rucio-37.0.0rc1.data/scripts/rucio +133 -0
- rucio-37.0.0rc1.data/scripts/rucio-abacus-account +74 -0
- rucio-37.0.0rc1.data/scripts/rucio-abacus-collection-replica +46 -0
- rucio-37.0.0rc1.data/scripts/rucio-abacus-rse +78 -0
- rucio-37.0.0rc1.data/scripts/rucio-admin +97 -0
- rucio-37.0.0rc1.data/scripts/rucio-atropos +60 -0
- rucio-37.0.0rc1.data/scripts/rucio-auditor +206 -0
- rucio-37.0.0rc1.data/scripts/rucio-automatix +50 -0
- rucio-37.0.0rc1.data/scripts/rucio-bb8 +57 -0
- rucio-37.0.0rc1.data/scripts/rucio-cache-client +141 -0
- rucio-37.0.0rc1.data/scripts/rucio-cache-consumer +42 -0
- rucio-37.0.0rc1.data/scripts/rucio-conveyor-finisher +58 -0
- rucio-37.0.0rc1.data/scripts/rucio-conveyor-poller +66 -0
- rucio-37.0.0rc1.data/scripts/rucio-conveyor-preparer +37 -0
- rucio-37.0.0rc1.data/scripts/rucio-conveyor-receiver +44 -0
- rucio-37.0.0rc1.data/scripts/rucio-conveyor-stager +76 -0
- rucio-37.0.0rc1.data/scripts/rucio-conveyor-submitter +139 -0
- rucio-37.0.0rc1.data/scripts/rucio-conveyor-throttler +104 -0
- rucio-37.0.0rc1.data/scripts/rucio-dark-reaper +53 -0
- rucio-37.0.0rc1.data/scripts/rucio-dumper +160 -0
- rucio-37.0.0rc1.data/scripts/rucio-follower +44 -0
- rucio-37.0.0rc1.data/scripts/rucio-hermes +54 -0
- rucio-37.0.0rc1.data/scripts/rucio-judge-cleaner +89 -0
- rucio-37.0.0rc1.data/scripts/rucio-judge-evaluator +137 -0
- rucio-37.0.0rc1.data/scripts/rucio-judge-injector +44 -0
- rucio-37.0.0rc1.data/scripts/rucio-judge-repairer +44 -0
- rucio-37.0.0rc1.data/scripts/rucio-kronos +44 -0
- rucio-37.0.0rc1.data/scripts/rucio-minos +53 -0
- rucio-37.0.0rc1.data/scripts/rucio-minos-temporary-expiration +50 -0
- rucio-37.0.0rc1.data/scripts/rucio-necromancer +120 -0
- rucio-37.0.0rc1.data/scripts/rucio-oauth-manager +63 -0
- rucio-37.0.0rc1.data/scripts/rucio-reaper +83 -0
- rucio-37.0.0rc1.data/scripts/rucio-replica-recoverer +248 -0
- rucio-37.0.0rc1.data/scripts/rucio-rse-decommissioner +66 -0
- rucio-37.0.0rc1.data/scripts/rucio-storage-consistency-actions +74 -0
- rucio-37.0.0rc1.data/scripts/rucio-transmogrifier +77 -0
- rucio-37.0.0rc1.data/scripts/rucio-undertaker +76 -0
- rucio-37.0.0rc1.dist-info/METADATA +92 -0
- rucio-37.0.0rc1.dist-info/RECORD +487 -0
- rucio-37.0.0rc1.dist-info/WHEEL +5 -0
- rucio-37.0.0rc1.dist-info/licenses/AUTHORS.rst +100 -0
- rucio-37.0.0rc1.dist-info/licenses/LICENSE +201 -0
- rucio-37.0.0rc1.dist-info/top_level.txt +1 -0
|
@@ -0,0 +1,13 @@
|
|
|
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.
|
|
@@ -0,0 +1,626 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
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
|
+
Suspicious-Replica-Recoverer is a daemon that deals with suspicious replicas based on if they are found available on other RSEs
|
|
18
|
+
or if they are the last remaining copy, on how many suspicious replicas are on a given RSE and on a replica's metadata.
|
|
19
|
+
Consequently, automatic replica recovery is triggered via necromancer daemon, which creates a rule for such bad replica(s).
|
|
20
|
+
"""
|
|
21
|
+
|
|
22
|
+
import functools
|
|
23
|
+
import json
|
|
24
|
+
import logging
|
|
25
|
+
import re
|
|
26
|
+
import threading
|
|
27
|
+
import time
|
|
28
|
+
from configparser import NoOptionError, NoSectionError
|
|
29
|
+
from datetime import datetime, timedelta
|
|
30
|
+
from typing import TYPE_CHECKING, Any, Optional
|
|
31
|
+
|
|
32
|
+
import rucio.db.sqla.util
|
|
33
|
+
from rucio.common.config import config_get, config_get_bool
|
|
34
|
+
from rucio.common.constants import SuspiciousAvailability
|
|
35
|
+
from rucio.common.exception import DatabaseException, DuplicateRule, VONotFound
|
|
36
|
+
from rucio.common.logging import setup_logging
|
|
37
|
+
from rucio.common.types import InternalAccount, LoggerFunction
|
|
38
|
+
from rucio.core.did import get_metadata
|
|
39
|
+
from rucio.core.replica import (
|
|
40
|
+
add_bad_pfns,
|
|
41
|
+
declare_bad_file_replicas,
|
|
42
|
+
get_suspicious_files,
|
|
43
|
+
get_suspicious_reason,
|
|
44
|
+
list_replicas,
|
|
45
|
+
)
|
|
46
|
+
from rucio.core.rse_expression_parser import parse_expression
|
|
47
|
+
from rucio.core.rule import add_rule
|
|
48
|
+
from rucio.core.vo import list_vos
|
|
49
|
+
from rucio.daemons.common import run_daemon
|
|
50
|
+
from rucio.db.sqla.util import get_db_time
|
|
51
|
+
|
|
52
|
+
if TYPE_CHECKING:
|
|
53
|
+
from collections.abc import Sequence
|
|
54
|
+
from types import FrameType
|
|
55
|
+
|
|
56
|
+
GRACEFUL_STOP = threading.Event()
|
|
57
|
+
DAEMON_NAME = 'suspicious-replica-recoverer'
|
|
58
|
+
|
|
59
|
+
|
|
60
|
+
def check_suspicious_policy(policy: dict[str, str], file_metadata_datatype: str, file_metadata_scope: str) -> str:
|
|
61
|
+
match_scope = False
|
|
62
|
+
match_datatype = False
|
|
63
|
+
action = ""
|
|
64
|
+
|
|
65
|
+
if not policy.get("scope", []):
|
|
66
|
+
match_scope = True
|
|
67
|
+
for scope in policy.get("scope", []):
|
|
68
|
+
if re.match(scope, file_metadata_scope):
|
|
69
|
+
match_scope = True
|
|
70
|
+
break
|
|
71
|
+
|
|
72
|
+
if not policy.get("datatype", []):
|
|
73
|
+
match_datatype = True
|
|
74
|
+
for datatype in policy.get("datatype", []):
|
|
75
|
+
if re.match(datatype, file_metadata_datatype):
|
|
76
|
+
match_datatype = True
|
|
77
|
+
break
|
|
78
|
+
|
|
79
|
+
if match_scope and match_datatype:
|
|
80
|
+
action = policy["action"]
|
|
81
|
+
|
|
82
|
+
return action
|
|
83
|
+
|
|
84
|
+
|
|
85
|
+
def parse_replica_datatype(
|
|
86
|
+
did_name_expression: Optional[str],
|
|
87
|
+
file_name: str,
|
|
88
|
+
rse_key: str,
|
|
89
|
+
logger: LoggerFunction) -> Optional[str]:
|
|
90
|
+
|
|
91
|
+
file_metadata_datatype = None
|
|
92
|
+
if did_name_expression is not None:
|
|
93
|
+
logger(logging.WARNING, f'Using {did_name_expression} as regex to determine file datatype for file: {file_name} on rse: {rse_key}')
|
|
94
|
+
try:
|
|
95
|
+
file_metadata_datatype_reg = re.compile(did_name_expression).search(file_name)
|
|
96
|
+
if file_metadata_datatype_reg is not None:
|
|
97
|
+
file_metadata_datatype = file_metadata_datatype_reg[0]
|
|
98
|
+
except (re.error, TypeError) as e:
|
|
99
|
+
logger(logging.WARNING, f'Could not determine datatype using regex {did_name_expression} on file: {file_name}, ignoring: {e}')
|
|
100
|
+
|
|
101
|
+
return file_metadata_datatype
|
|
102
|
+
|
|
103
|
+
|
|
104
|
+
def _read_from_config(logger: LoggerFunction) -> tuple[str, bool, Optional[str]]:
|
|
105
|
+
rule_suspicious_rse_expression = config_get("replicarecoverer", "rule_rse_expression", raise_exception=False, default='type=SCRATCHDISK')
|
|
106
|
+
logger(logging.INFO, f"Recovering with the rse expression: {rule_suspicious_rse_expression}")
|
|
107
|
+
|
|
108
|
+
use_file_metadata = config_get_bool("replicarecoverer", "use_file_metadata", raise_exception=False, default=True)
|
|
109
|
+
did_name_expression = None
|
|
110
|
+
|
|
111
|
+
if not use_file_metadata:
|
|
112
|
+
try:
|
|
113
|
+
did_name_expression = config_get('replicarecoverer', 'did_name_expression', raise_exception=True)
|
|
114
|
+
logger(logging.INFO, f"Setting replica datatype from did name with regex expression: {did_name_expression}")
|
|
115
|
+
|
|
116
|
+
except (NoOptionError, NoSectionError) as e:
|
|
117
|
+
logger(logging.WARNING, f"Cannot use option 'use_file_metadata' without setting a replacement expression with 'did_name_expression'- {e}")
|
|
118
|
+
use_file_metadata = True
|
|
119
|
+
|
|
120
|
+
logger(logging.INFO, f"Recovering with the rse expression: {rule_suspicious_rse_expression}")
|
|
121
|
+
return rule_suspicious_rse_expression, use_file_metadata, did_name_expression
|
|
122
|
+
|
|
123
|
+
|
|
124
|
+
def declare_suspicious_replicas_bad(
|
|
125
|
+
once: bool = False,
|
|
126
|
+
younger_than: int = 5,
|
|
127
|
+
nattempts: int = 5,
|
|
128
|
+
vos: "Optional[Sequence[str]]" = None,
|
|
129
|
+
limit_suspicious_files_on_rse: int = 5,
|
|
130
|
+
json_file_name: str = "/opt/rucio/etc/suspicious_replica_recoverer.json",
|
|
131
|
+
sleep_time: int = 3600,
|
|
132
|
+
active_mode: bool = False) -> None:
|
|
133
|
+
"""
|
|
134
|
+
Main loop to check for available replicas which are labeled as suspicious.
|
|
135
|
+
|
|
136
|
+
Gets a list of suspicious replicas that are listed as AVAILABLE in 'replicas' table
|
|
137
|
+
and available on other RSE. Finds pfns of these replicas and declares them as bad.
|
|
138
|
+
Replicas that are the last remaining copy of a file have additional checks (checksum
|
|
139
|
+
comparison, etc.) before being declared bad.
|
|
140
|
+
|
|
141
|
+
:param once: If True, the loop is run just once, otherwise the daemon continues looping until stopped.
|
|
142
|
+
:param younger_than: The number of days since which bad_replicas table will be searched
|
|
143
|
+
for finding replicas declared 'SUSPICIOUS' at a specific RSE ('rse_expression'),
|
|
144
|
+
but 'AVAILABLE' on other RSE(s).
|
|
145
|
+
:param nattempts: The minimum number of appearances in the bad_replica DB table
|
|
146
|
+
in order to appear in the resulting list of replicas for recovery.
|
|
147
|
+
:param vos: VOs on which to look for RSEs. Only used in multi-VO mode.
|
|
148
|
+
If empty, we either use all VOs if run from "def",
|
|
149
|
+
:param limit_suspicious_files_on_rse: Maximum number of suspicious replicas on an RSE before that RSE
|
|
150
|
+
is considered problematic and the suspicious replicas on that RSE
|
|
151
|
+
are labeled as 'TEMPORARY_UNAVAILABLE'.
|
|
152
|
+
:param sleep_time: The daemon should not run too often. If the daemon's runtime is quicker than sleep_time, then
|
|
153
|
+
it should sleep until sleep_time is over.
|
|
154
|
+
:returns: None
|
|
155
|
+
"""
|
|
156
|
+
|
|
157
|
+
run_daemon(
|
|
158
|
+
once=once,
|
|
159
|
+
graceful_stop=GRACEFUL_STOP,
|
|
160
|
+
executable=DAEMON_NAME,
|
|
161
|
+
partition_wait_time=1,
|
|
162
|
+
sleep_time=sleep_time,
|
|
163
|
+
run_once_fnc=functools.partial(
|
|
164
|
+
run_once,
|
|
165
|
+
younger_than=younger_than,
|
|
166
|
+
nattempts=nattempts,
|
|
167
|
+
vos=vos,
|
|
168
|
+
limit_suspicious_files_on_rse=limit_suspicious_files_on_rse,
|
|
169
|
+
json_file_name=json_file_name,
|
|
170
|
+
active_mode=active_mode
|
|
171
|
+
),
|
|
172
|
+
)
|
|
173
|
+
|
|
174
|
+
|
|
175
|
+
def run_once(heartbeat_handler: Any, younger_than: int, nattempts: int, vos: "Optional[Sequence[str]]", limit_suspicious_files_on_rse: int, json_file_name: str, active_mode: int, **_kwargs) -> bool:
|
|
176
|
+
|
|
177
|
+
_, _, logger = heartbeat_handler.live()
|
|
178
|
+
rule_suspicious_rse_expression, use_file_metadata, did_name_expression = _read_from_config(logger=logger)
|
|
179
|
+
vos = vos or []
|
|
180
|
+
|
|
181
|
+
multi_vo = config_get_bool('common', 'multi_vo', raise_exception=False, default=False)
|
|
182
|
+
if not multi_vo:
|
|
183
|
+
if vos:
|
|
184
|
+
logger(logging.WARNING, 'Ignoring argument vos, this is only applicable in a multi-VO setup.')
|
|
185
|
+
vos = ['def']
|
|
186
|
+
else:
|
|
187
|
+
if vos:
|
|
188
|
+
invalid = set(vos) - set([v['vo'] for v in list_vos()])
|
|
189
|
+
if invalid:
|
|
190
|
+
msg = 'VO{} {} cannot be found'.format('s' if len(invalid) > 1 else '', ', '.join([repr(v) for v in invalid]))
|
|
191
|
+
raise VONotFound(msg)
|
|
192
|
+
else:
|
|
193
|
+
vos = [v['vo'] for v in list_vos()]
|
|
194
|
+
logger(logging.INFO, 'This instance will work on VO%s: %s' % ('s' if len(vos) > 1 else '', ', '.join([v for v in vos])))
|
|
195
|
+
|
|
196
|
+
start = time.time()
|
|
197
|
+
|
|
198
|
+
try:
|
|
199
|
+
json_file = open(json_file_name, mode="r")
|
|
200
|
+
logger(logging.INFO, "JSON file has been opened.")
|
|
201
|
+
except:
|
|
202
|
+
logger(logging.WARNING, "An error occurred while trying to open the JSON file.")
|
|
203
|
+
must_sleep = True
|
|
204
|
+
return must_sleep
|
|
205
|
+
|
|
206
|
+
try:
|
|
207
|
+
json_data = json.load(json_file)
|
|
208
|
+
except ValueError as e:
|
|
209
|
+
logger(logging.WARNING, "No JSON object could be decoded. Error: %s", e)
|
|
210
|
+
return True
|
|
211
|
+
|
|
212
|
+
# Checking that the json file is formatedd properly.
|
|
213
|
+
for i, entry in enumerate(json_data):
|
|
214
|
+
if ("datatype" not in entry or "scope" not in entry or "action" not in entry):
|
|
215
|
+
logger(logging.ERROR, 'Entry %s in the json file is incomplete (missing either "datatype" or "action").', i)
|
|
216
|
+
break
|
|
217
|
+
|
|
218
|
+
logger(logging.INFO, 'Ready to query replicas that were reported as suspicious in the last %s days at least %s times.', younger_than, nattempts)
|
|
219
|
+
|
|
220
|
+
getfileskwargs = {'younger_than': younger_than,
|
|
221
|
+
'nattempts': nattempts,
|
|
222
|
+
'exclude_states': ['B', 'R', 'D', 'L', 'T'],
|
|
223
|
+
'is_suspicious': True}
|
|
224
|
+
getfileskwargs_nattempts_1 = {'younger_than': younger_than,
|
|
225
|
+
'nattempts': 1,
|
|
226
|
+
'nattempts_exact': True,
|
|
227
|
+
'exclude_states': ['B', 'R', 'D', 'L', 'T'],
|
|
228
|
+
'is_suspicious': True}
|
|
229
|
+
|
|
230
|
+
for vo in vos:
|
|
231
|
+
logger(logging.INFO, 'Start replica recovery for VO: %s', vo)
|
|
232
|
+
recoverable_replicas = {}
|
|
233
|
+
if vo not in recoverable_replicas:
|
|
234
|
+
recoverable_replicas[vo] = {}
|
|
235
|
+
# Separate replicas that have only been declared suspicious once from the rest,
|
|
236
|
+
# as they will be handled differently and shouldn't be considered when deciding
|
|
237
|
+
# if an RSE is problematic (due to a high number of suspicious replicas)
|
|
238
|
+
replicas_nattempts_1 = {}
|
|
239
|
+
if vo not in replicas_nattempts_1:
|
|
240
|
+
replicas_nattempts_1[vo] = {}
|
|
241
|
+
|
|
242
|
+
rse_list = sorted([rse for rse in parse_expression('enable_suspicious_file_recovery=true') if rse['vo'] == vo], key=lambda k: k['rse'])
|
|
243
|
+
|
|
244
|
+
logger(logging.DEBUG, "List of RSEs with enable_suspicious_file_recovery = True: (total: %i)", len(rse_list))
|
|
245
|
+
for i in rse_list:
|
|
246
|
+
logger(logging.DEBUG, '%s', i)
|
|
247
|
+
|
|
248
|
+
for rse in rse_list:
|
|
249
|
+
time_start_rse = time.time()
|
|
250
|
+
rse_expr = rse['rse']
|
|
251
|
+
cnt_pfn_not_found = 0
|
|
252
|
+
cnt_pfn_not_found_nattempts_1 = 0
|
|
253
|
+
if rse_expr not in recoverable_replicas[vo]:
|
|
254
|
+
recoverable_replicas[vo][rse_expr] = {}
|
|
255
|
+
if rse_expr not in replicas_nattempts_1[vo]:
|
|
256
|
+
replicas_nattempts_1[vo][rse_expr] = {}
|
|
257
|
+
# Get a dictionary of the suspicious replicas on the RSE that have available copies on other RSEs
|
|
258
|
+
suspicious_replicas_avail_elsewhere = get_suspicious_files(rse_expr, available_elsewhere=SuspiciousAvailability["EXIST_COPIES"].value, filter_={'vo': vo}, **getfileskwargs)
|
|
259
|
+
# Get the suspicious replicas that are the last remaining copies
|
|
260
|
+
suspicious_replicas_last_copy = get_suspicious_files(rse_expr, available_elsewhere=SuspiciousAvailability["LAST_COPY"].value, filter_={'vo': vo}, **getfileskwargs)
|
|
261
|
+
# Get the suspicious replicas that have only been declared once
|
|
262
|
+
suspicious_replicas_nattempts_1 = get_suspicious_files(rse_expr, available_elsewhere=SuspiciousAvailability["ALL"].value, filter_={'vo': vo}, **getfileskwargs_nattempts_1)
|
|
263
|
+
|
|
264
|
+
logger(logging.DEBUG, 'Suspicious replicas on %s:', rse_expr)
|
|
265
|
+
logger(logging.DEBUG, 'Replicas with copies on other RSEs (%s):', len(suspicious_replicas_avail_elsewhere))
|
|
266
|
+
for i in suspicious_replicas_avail_elsewhere:
|
|
267
|
+
logger(logging.DEBUG, '%s', i)
|
|
268
|
+
logger(logging.DEBUG, 'Replicas that are the last remaining copy (%s):', len(suspicious_replicas_last_copy))
|
|
269
|
+
for i in suspicious_replicas_last_copy:
|
|
270
|
+
logger(logging.DEBUG, '%s', i)
|
|
271
|
+
logger(logging.DEBUG, 'Replicas that have only been declared once (%s):', len(suspicious_replicas_nattempts_1))
|
|
272
|
+
for i in suspicious_replicas_nattempts_1:
|
|
273
|
+
logger(logging.DEBUG, '%s', i)
|
|
274
|
+
|
|
275
|
+
# RSEs that aren't available shouldn't have suspicious replicas showing up. Skip to next RSE.
|
|
276
|
+
if not rse['availability_read'] and ((len(suspicious_replicas_avail_elsewhere) > 0) or (len(suspicious_replicas_last_copy) > 0)):
|
|
277
|
+
logger(logging.WARNING, "%s is not available (availability: %s), yet it has suspicious replicas. Please investigate. \n", rse_expr, rse['availability_read'])
|
|
278
|
+
continue
|
|
279
|
+
|
|
280
|
+
if suspicious_replicas_avail_elsewhere:
|
|
281
|
+
for replica in suspicious_replicas_avail_elsewhere:
|
|
282
|
+
if vo == replica['scope'].vo:
|
|
283
|
+
scope = replica['scope']
|
|
284
|
+
rep_name = replica['name']
|
|
285
|
+
rse_id = replica['rse_id']
|
|
286
|
+
pfn_not_found = True
|
|
287
|
+
for rep in list_replicas([{'scope': scope, 'name': rep_name}]):
|
|
288
|
+
for rse_ in rep['rses']:
|
|
289
|
+
if rse_ == rse_id:
|
|
290
|
+
recoverable_replicas[vo][rse_expr][rep_name] = {'name': rep_name, 'rse_id': rse_id, 'scope': scope, 'pfn': rep['rses'][rse_][0], 'available_elsewhere': True}
|
|
291
|
+
pfn_not_found = False
|
|
292
|
+
|
|
293
|
+
if pfn_not_found:
|
|
294
|
+
cnt_pfn_not_found += 1
|
|
295
|
+
logger(logging.WARNING, 'Skipping suspicious replica %s on %s, no pfns were found.', rep_name, rse_expr)
|
|
296
|
+
|
|
297
|
+
if suspicious_replicas_last_copy:
|
|
298
|
+
for replica in suspicious_replicas_last_copy:
|
|
299
|
+
if vo == replica['scope'].vo:
|
|
300
|
+
scope = replica['scope']
|
|
301
|
+
rep_name = replica['name']
|
|
302
|
+
rse_id = replica['rse_id']
|
|
303
|
+
pfn_not_found = True
|
|
304
|
+
# Should only return one rse, as there is only one replica remaining
|
|
305
|
+
for rep in list_replicas([{'scope': scope, 'name': rep_name}]):
|
|
306
|
+
recoverable_replicas[vo][rse_expr][rep_name] = {'name': rep_name, 'rse_id': rse_id, 'scope': scope, 'pfn': rep['rses'][rse_id][0], 'available_elsewhere': False}
|
|
307
|
+
pfn_not_found = False
|
|
308
|
+
if pfn_not_found:
|
|
309
|
+
cnt_pfn_not_found += 1
|
|
310
|
+
logger(logging.WARNING, 'Skipping suspicious replica %s on %s, no pfns were found.', rep_name, rse_expr)
|
|
311
|
+
|
|
312
|
+
if suspicious_replicas_nattempts_1:
|
|
313
|
+
for replica in suspicious_replicas_nattempts_1:
|
|
314
|
+
if vo == replica['scope'].vo:
|
|
315
|
+
scope = replica['scope']
|
|
316
|
+
rep_name = replica['name']
|
|
317
|
+
rse_id = replica['rse_id']
|
|
318
|
+
pfn_not_found = True
|
|
319
|
+
for rep in list_replicas([{'scope': scope, 'name': rep_name}]):
|
|
320
|
+
for rse_ in rep['rses']:
|
|
321
|
+
if rse_ == rse_id:
|
|
322
|
+
replicas_nattempts_1[vo][rse_expr][rep_name] = {'name': rep_name, 'rse_id': rse_id, 'scope': scope, 'pfn': rep['rses'][rse_][0], 'available_elsewhere': True}
|
|
323
|
+
pfn_not_found = False
|
|
324
|
+
if pfn_not_found:
|
|
325
|
+
cnt_pfn_not_found_nattempts_1 += 1
|
|
326
|
+
logger(logging.WARNING, 'Skipping suspicious replica %s on %s, no pfns were found.', rep_name, rse_expr)
|
|
327
|
+
|
|
328
|
+
logger(logging.INFO, 'Suspicious replica query took %s seconds on %s and found %i suspicious replica(s) with a minimum of nattempts=%i. The pfns for %s/%s replicas were found.',
|
|
329
|
+
time.time() - time_start_rse,
|
|
330
|
+
rse_expr,
|
|
331
|
+
len(suspicious_replicas_avail_elsewhere) + len(suspicious_replicas_last_copy),
|
|
332
|
+
nattempts,
|
|
333
|
+
len(suspicious_replicas_avail_elsewhere) + len(suspicious_replicas_last_copy) - cnt_pfn_not_found,
|
|
334
|
+
len(suspicious_replicas_avail_elsewhere) + len(suspicious_replicas_last_copy))
|
|
335
|
+
|
|
336
|
+
logger(logging.INFO, 'A total of %i replicas with exactly nattempts=1 were found. The pfns for %s/%s replicas were found.',
|
|
337
|
+
len(suspicious_replicas_nattempts_1),
|
|
338
|
+
len(suspicious_replicas_nattempts_1) - cnt_pfn_not_found_nattempts_1,
|
|
339
|
+
len(suspicious_replicas_nattempts_1))
|
|
340
|
+
|
|
341
|
+
if len(suspicious_replicas_avail_elsewhere) + len(suspicious_replicas_last_copy) != 0:
|
|
342
|
+
logger(logging.DEBUG, 'List of replicas on %s for which the pfns have been found:', rse_expr)
|
|
343
|
+
for i in recoverable_replicas[vo][rse_expr]:
|
|
344
|
+
logger(logging.DEBUG, '%s', i)
|
|
345
|
+
|
|
346
|
+
if len(suspicious_replicas_nattempts_1) != 0:
|
|
347
|
+
logger(logging.DEBUG, 'List of replicas on %s with nattempts=1 for which the pfns have been found:', rse_expr)
|
|
348
|
+
for i in replicas_nattempts_1[vo][rse_expr]:
|
|
349
|
+
logger(logging.DEBUG, '%s', i)
|
|
350
|
+
|
|
351
|
+
logger(logging.INFO, 'All RSEs have been checked for suspicious replicas. Total time: %s seconds.', time.time() - start)
|
|
352
|
+
|
|
353
|
+
# Checking that everything is still working properly
|
|
354
|
+
_, _, logger = heartbeat_handler.live()
|
|
355
|
+
|
|
356
|
+
logger(logging.INFO, 'Create rules for replicas with nattempts=1.')
|
|
357
|
+
|
|
358
|
+
# Create as many rules as necessary for the replicas to be picked up by the daemon on the next run
|
|
359
|
+
# Create rules only for replicas that can be declared bad.
|
|
360
|
+
# Replicas from the auditor should be declared bad regardless of suspicious declarations, so no rules necessary.
|
|
361
|
+
for rse_key in list(replicas_nattempts_1[vo].keys()):
|
|
362
|
+
if not replicas_nattempts_1[vo][rse_key]:
|
|
363
|
+
# This is needed for testing purposes.
|
|
364
|
+
continue
|
|
365
|
+
files_to_be_declared_bad_nattempts_1 = []
|
|
366
|
+
dids_nattempts_1 = []
|
|
367
|
+
# Get the rse_id by going to one of the suspicious replicas from that RSE and reading it from there
|
|
368
|
+
rse_id = list(replicas_nattempts_1[vo][rse_key].values())[0]['rse_id']
|
|
369
|
+
for replica_key in replicas_nattempts_1[vo][rse_key].keys():
|
|
370
|
+
from_auditor = False
|
|
371
|
+
file_scope = replicas_nattempts_1[vo][rse_key][replica_key]["scope"]
|
|
372
|
+
file_name = replicas_nattempts_1[vo][rse_key][replica_key]["name"]
|
|
373
|
+
file_metadata = get_metadata(file_scope, file_name)
|
|
374
|
+
replicas_nattempts_1[vo][rse_key][replica_key]["datatype"] = str(file_metadata["datatype"])
|
|
375
|
+
|
|
376
|
+
# Auditor
|
|
377
|
+
suspicious_reason = get_suspicious_reason(replicas_nattempts_1[vo][rse_key][replica_key]["rse_id"], file_scope, file_name, nattempts)
|
|
378
|
+
for reason in suspicious_reason:
|
|
379
|
+
if "auditor" in reason["reason"].lower():
|
|
380
|
+
from_auditor = True
|
|
381
|
+
files_to_be_declared_bad_nattempts_1.append(replicas_nattempts_1[vo][rse_key][replica_key])
|
|
382
|
+
break
|
|
383
|
+
|
|
384
|
+
# Bad
|
|
385
|
+
if not from_auditor:
|
|
386
|
+
if (file_name.startswith("log.")) or (file_name.startswith("user")):
|
|
387
|
+
# Don't keep log files or user files
|
|
388
|
+
files_to_be_declared_bad_nattempts_1.append(replicas_nattempts_1[vo][rse_key][replica_key])
|
|
389
|
+
action = ""
|
|
390
|
+
else:
|
|
391
|
+
# Deal with replicas based on their metadata.
|
|
392
|
+
if (file_metadata["datatype"] is None) and (use_file_metadata): # "None" type has no function "split()"
|
|
393
|
+
logger(logging.WARNING, "RSE: %s, replica name %s, pfn %s: Replica does not have a data type associated with it. No action will be taken.",
|
|
394
|
+
rse_key, replica_key, replicas_nattempts_1[vo][rse_key][replica_key]['pfn'])
|
|
395
|
+
continue
|
|
396
|
+
file_metadata_datatype = str(file_metadata["datatype"])
|
|
397
|
+
if not use_file_metadata:
|
|
398
|
+
file_metadata_datatype = parse_replica_datatype(did_name_expression, file_name, rse_key, logger)
|
|
399
|
+
file_metadata_scope = str(file_metadata["scope"])
|
|
400
|
+
action = ""
|
|
401
|
+
if file_metadata_datatype:
|
|
402
|
+
# Some files don't have a datatype. They should be ignored.
|
|
403
|
+
for policy in json_data:
|
|
404
|
+
action = check_suspicious_policy(policy=policy, file_metadata_datatype=file_metadata_datatype, file_metadata_scope=file_metadata_scope)
|
|
405
|
+
if action:
|
|
406
|
+
logger(logging.INFO, "The action that will be performed is %s", action)
|
|
407
|
+
break
|
|
408
|
+
if action:
|
|
409
|
+
# Rules will be created for these replicas.
|
|
410
|
+
dids = {'scope': file_scope, 'name': file_name, 'rse': rse_key}
|
|
411
|
+
dids_nattempts_1.append(dids)
|
|
412
|
+
if active_mode:
|
|
413
|
+
if len(dids_nattempts_1) > 0:
|
|
414
|
+
try:
|
|
415
|
+
add_rule(dids=dids_nattempts_1, account=InternalAccount('root', vo=vo), copies=nattempts, rse_expression=rule_suspicious_rse_expression, grouping=None, weight=None, lifetime=5 * 24 * 3600, locked=False, subscription_id=None)
|
|
416
|
+
logger(logging.INFO, 'Rules have been created for %i replicas on %s.', len(dids_nattempts_1), rse_key)
|
|
417
|
+
except DuplicateRule:
|
|
418
|
+
logger(logging.INFO, 'Tried to create rules on %s, but it already exists.', rse_key)
|
|
419
|
+
else:
|
|
420
|
+
logger(logging.INFO, 'No replicas on %s have nattmepts=1, so no rules have been created.', rse_key)
|
|
421
|
+
if len(files_to_be_declared_bad_nattempts_1) > 0:
|
|
422
|
+
logger(logging.INFO, 'Ready to declare %s bad replica(s) with nattempts=1 on %s (RSE id: %s).', len(files_to_be_declared_bad_nattempts_1), rse_key, str(rse_id))
|
|
423
|
+
declare_bad_file_replicas(replicas=files_to_be_declared_bad_nattempts_1, reason='Suspicious. Automatic recovery.', issuer=InternalAccount('root', vo=vo), session=None)
|
|
424
|
+
else:
|
|
425
|
+
logger(logging.INFO, 'No suspicious replica(s) with nattempts=1 on %s (RSE id: %s) have been declared bad.', rse_key, str(rse_id))
|
|
426
|
+
else:
|
|
427
|
+
logger(logging.INFO, 'No replicas on %s with nattempts=1.', rse_key)
|
|
428
|
+
|
|
429
|
+
logger(logging.INFO, 'Begin check for problematic RSEs.')
|
|
430
|
+
time_start_check_probl = time.time()
|
|
431
|
+
|
|
432
|
+
# If an RSE has more than *limit_suspicious_files_on_rse* suspicious files, then there might be a problem with the RSE.
|
|
433
|
+
# The suspicious files are marked as temporarily unavailable.
|
|
434
|
+
list_problematic_rses = []
|
|
435
|
+
for rse_key in list(recoverable_replicas[vo].keys()):
|
|
436
|
+
if len(recoverable_replicas[vo][rse_key].values()) > limit_suspicious_files_on_rse:
|
|
437
|
+
list_problematic_rses.append(rse_key)
|
|
438
|
+
pfns_list = []
|
|
439
|
+
for replica_value in recoverable_replicas[vo][rse_key].values():
|
|
440
|
+
pfns_list.append(replica_value['pfn'])
|
|
441
|
+
|
|
442
|
+
if active_mode:
|
|
443
|
+
add_bad_pfns(pfns=pfns_list, account=InternalAccount('root', vo=vo), state='TEMPORARY_UNAVAILABLE', expires_at=datetime.utcnow() + timedelta(days=3))
|
|
444
|
+
|
|
445
|
+
logger(logging.INFO, "%s is problematic (more than %s suspicious replicas). Send a Jira ticket for the RSE (to be implemented).", rse_key, limit_suspicious_files_on_rse)
|
|
446
|
+
logger(logging.INFO, "The following files on %s have been marked as TEMPORARILY UNAVAILABLE:", rse_key)
|
|
447
|
+
for replica_values in recoverable_replicas[vo][rse_key].values():
|
|
448
|
+
logger(logging.INFO, 'Temporarily unavailable: RSE: %s Scope: %s Name: %s PFN: %s', rse_key, replica_values['scope'], replica_values['name'], replica_values['pfn'])
|
|
449
|
+
# Remove the RSE from the dictionary as it has been dealt with.
|
|
450
|
+
del recoverable_replicas[vo][rse_key]
|
|
451
|
+
|
|
452
|
+
logger(logging.INFO, "Following RSEs were deemed problematic (total: %s)", len(list_problematic_rses))
|
|
453
|
+
for rse in list_problematic_rses:
|
|
454
|
+
logger(logging.INFO, "%s", rse)
|
|
455
|
+
|
|
456
|
+
# Checking that everything is still working properly
|
|
457
|
+
_, _, logger = heartbeat_handler.live()
|
|
458
|
+
|
|
459
|
+
auditor = 0
|
|
460
|
+
checksum = 0
|
|
461
|
+
|
|
462
|
+
# Label suspicious replicas as bad if they have other copies on other RSEs (that aren't also marked as suspicious).
|
|
463
|
+
# If they are the last remaining copies, deal with them differently.
|
|
464
|
+
for rse_key in list(recoverable_replicas[vo].keys()):
|
|
465
|
+
files_to_be_declared_bad = []
|
|
466
|
+
files_to_be_ignored = []
|
|
467
|
+
files_dry_run_monitoring = []
|
|
468
|
+
# Remove RSEs from dictionary that don't have any suspicious replicas
|
|
469
|
+
if len(recoverable_replicas[vo][rse_key]) == 0:
|
|
470
|
+
del recoverable_replicas[vo][rse_key]
|
|
471
|
+
continue
|
|
472
|
+
# Get the rse_id by going to one of the suspicious replicas from that RSE and reading it from there
|
|
473
|
+
rse_id = list(recoverable_replicas[vo][rse_key].values())[0]['rse_id']
|
|
474
|
+
for replica_key in list(recoverable_replicas[vo][rse_key].keys()):
|
|
475
|
+
from_auditor = False
|
|
476
|
+
file_scope = recoverable_replicas[vo][rse_key][replica_key]["scope"]
|
|
477
|
+
file_name = recoverable_replicas[vo][rse_key][replica_key]["name"]
|
|
478
|
+
file_metadata = get_metadata(file_scope, file_name)
|
|
479
|
+
recoverable_replicas[vo][rse_key][replica_key]["datatype"] = str(file_metadata["datatype"])
|
|
480
|
+
|
|
481
|
+
suspicious_reason = get_suspicious_reason(recoverable_replicas[vo][rse_key][replica_key]["rse_id"],
|
|
482
|
+
file_scope,
|
|
483
|
+
file_name,
|
|
484
|
+
nattempts)
|
|
485
|
+
for reason in suspicious_reason:
|
|
486
|
+
if "auditor" in reason["reason"].lower():
|
|
487
|
+
auditor += 1
|
|
488
|
+
files_to_be_declared_bad.append(recoverable_replicas[vo][rse_key][replica_key])
|
|
489
|
+
from_auditor = True
|
|
490
|
+
break
|
|
491
|
+
|
|
492
|
+
if not from_auditor:
|
|
493
|
+
if recoverable_replicas[vo][rse_key][replica_key]['available_elsewhere'] is True:
|
|
494
|
+
# Replicas with other copies on at least one other RSE can safely be labeled as bad
|
|
495
|
+
files_to_be_declared_bad.append(recoverable_replicas[vo][rse_key][replica_key])
|
|
496
|
+
# Remove replica from dictionary
|
|
497
|
+
del recoverable_replicas[vo][rse_key][replica_key]
|
|
498
|
+
elif recoverable_replicas[vo][rse_key][replica_key]['available_elsewhere'] is False:
|
|
499
|
+
if (file_name.startswith("log.")) or (file_name.startswith("user")):
|
|
500
|
+
# Don't keep log files or user files
|
|
501
|
+
files_to_be_declared_bad.append(recoverable_replicas[vo][rse_key][replica_key])
|
|
502
|
+
del recoverable_replicas[vo][rse_key][replica_key]
|
|
503
|
+
else:
|
|
504
|
+
# Deal with replicas based on their metadata.
|
|
505
|
+
if (file_metadata["datatype"] is None) and use_file_metadata: # "None" type has no function "split()"
|
|
506
|
+
files_to_be_ignored.append(recoverable_replicas[vo][rse_key][replica_key])
|
|
507
|
+
logger(logging.WARNING, "RSE: %s, replica name %s, pfn %s: Replica does not have a data type associated with it. No action will be taken.",
|
|
508
|
+
rse_key, replica_key, recoverable_replicas[vo][rse_key][replica_key]['pfn'])
|
|
509
|
+
continue
|
|
510
|
+
|
|
511
|
+
file_metadata_datatype = str(file_metadata["datatype"])
|
|
512
|
+
if not use_file_metadata:
|
|
513
|
+
file_metadata_datatype = parse_replica_datatype(did_name_expression, file_name, rse_key, logger)
|
|
514
|
+
file_metadata_scope = str(file_metadata["scope"])
|
|
515
|
+
action = ""
|
|
516
|
+
if file_metadata_datatype:
|
|
517
|
+
# Some files don't have a datatype. They should be ignored.
|
|
518
|
+
for policy in json_data:
|
|
519
|
+
action = check_suspicious_policy(policy=policy, file_metadata_datatype=file_metadata_datatype, file_metadata_scope=file_metadata_scope)
|
|
520
|
+
if action:
|
|
521
|
+
logger(logging.INFO, "The action that will be performed is %s", action)
|
|
522
|
+
break
|
|
523
|
+
|
|
524
|
+
if not action:
|
|
525
|
+
logger(logging.WARNING, "No recognised actions (ignore/declare bad) found in policy file (etc/suspicious_replica_recoverer.json). Replica will be ignored by default.")
|
|
526
|
+
|
|
527
|
+
if action:
|
|
528
|
+
if action == "dry run":
|
|
529
|
+
# Monitoring purposes: Will look like a file has been declared bad, even though no
|
|
530
|
+
# actions will be taken.
|
|
531
|
+
files_dry_run_monitoring.append(recoverable_replicas[vo][rse_key][replica_key])
|
|
532
|
+
elif action == "ignore":
|
|
533
|
+
files_to_be_ignored.append(recoverable_replicas[vo][rse_key][replica_key])
|
|
534
|
+
elif action == "declare bad":
|
|
535
|
+
suspicious_reason = get_suspicious_reason(recoverable_replicas[vo][rse_key][replica_key]["rse_id"],
|
|
536
|
+
file_scope,
|
|
537
|
+
file_name,
|
|
538
|
+
nattempts)
|
|
539
|
+
for reason in suspicious_reason:
|
|
540
|
+
if "checksum" in reason["reason"].lower():
|
|
541
|
+
checksum += 1
|
|
542
|
+
files_to_be_declared_bad.append(recoverable_replicas[vo][rse_key][replica_key])
|
|
543
|
+
break
|
|
544
|
+
else:
|
|
545
|
+
# If no policy has been set, default to ignoring the file (no action taken).
|
|
546
|
+
files_to_be_ignored.append(recoverable_replicas[vo][rse_key][replica_key])
|
|
547
|
+
|
|
548
|
+
logger(logging.INFO, '(%s) Remaining replicas (pfns) that will be ignored:', rse_key)
|
|
549
|
+
for i in files_to_be_ignored:
|
|
550
|
+
logger(logging.INFO, 'Ignore: RSE: %s Scope: %s Name: %s Datatype: %s PFN: %s', rse_key, i["scope"], i["name"], i["datatype"], i["pfn"])
|
|
551
|
+
logger(logging.INFO, '(%s) Remaining replica (pfns) that will be declared BAD:', rse_key)
|
|
552
|
+
for i in files_to_be_declared_bad:
|
|
553
|
+
logger(logging.INFO, 'Declare bad: RSE: %s Scope: %s Name: %s Datatype: %s PFN: %s', rse_key, i["scope"], i["name"], i["datatype"], i["pfn"])
|
|
554
|
+
for i in files_dry_run_monitoring:
|
|
555
|
+
logger(logging.INFO, 'Declare bad (dry run): RSE: %s Scope: %s Name: %s Datatype: %s PFN: %s', rse_key, i["scope"], i["name"], i["datatype"], i["pfn"])
|
|
556
|
+
|
|
557
|
+
if files_to_be_declared_bad:
|
|
558
|
+
logger(logging.INFO, 'Ready to declare %s bad replica(s) on %s (RSE id: %s).', len(files_to_be_declared_bad), rse_key, str(rse_id))
|
|
559
|
+
logger(logging.INFO, 'Number of replicas with checksum problems: %i', checksum)
|
|
560
|
+
logger(logging.INFO, 'Number of replicas that were declared suspicious by the auditor: %i', auditor)
|
|
561
|
+
|
|
562
|
+
if active_mode:
|
|
563
|
+
declare_bad_file_replicas(replicas=files_to_be_declared_bad, reason='Suspicious. Automatic recovery.', issuer=InternalAccount('root', vo=vo), session=None)
|
|
564
|
+
|
|
565
|
+
logger(logging.INFO, 'Finished declaring bad replicas on %s.\n', rse_key)
|
|
566
|
+
else:
|
|
567
|
+
logger(logging.INFO, 'No files were declared bad on %s.\n', rse_key)
|
|
568
|
+
|
|
569
|
+
logger(logging.INFO, 'Finished checking for problematic RSEs and declaring bad replicas on VO "%s". Total time: %s seconds.', vo, time.time() - time_start_check_probl)
|
|
570
|
+
|
|
571
|
+
time_passed = time.time() - start
|
|
572
|
+
logger(logging.INFO, 'Total time for VO "%s": %s seconds', vo, time_passed)
|
|
573
|
+
|
|
574
|
+
time_passed = time.time() - start
|
|
575
|
+
logger(logging.INFO, 'Total time: %s seconds', time_passed)
|
|
576
|
+
must_sleep = True
|
|
577
|
+
return must_sleep
|
|
578
|
+
|
|
579
|
+
|
|
580
|
+
def run(
|
|
581
|
+
once: bool = False,
|
|
582
|
+
younger_than: int = 5,
|
|
583
|
+
nattempts: int = 5,
|
|
584
|
+
vos: "Optional[Sequence[str]]" = None,
|
|
585
|
+
limit_suspicious_files_on_rse: int = 5,
|
|
586
|
+
json_file_name: str = "/opt/rucio/etc/suspicious_replica_recoverer.json",
|
|
587
|
+
sleep_time: int = 3600,
|
|
588
|
+
active_mode: bool = False) -> None:
|
|
589
|
+
"""
|
|
590
|
+
Starts up the Suspicious-Replica-Recoverer threads.
|
|
591
|
+
"""
|
|
592
|
+
setup_logging(process_name=DAEMON_NAME)
|
|
593
|
+
|
|
594
|
+
if rucio.db.sqla.util.is_old_db():
|
|
595
|
+
raise DatabaseException('Database was not updated, daemon won\'t start')
|
|
596
|
+
|
|
597
|
+
client_time, db_time = datetime.utcnow(), get_db_time()
|
|
598
|
+
max_offset = timedelta(hours=1, seconds=10)
|
|
599
|
+
if isinstance(db_time, datetime):
|
|
600
|
+
if db_time - client_time > max_offset or client_time - db_time > max_offset:
|
|
601
|
+
logging.critical('Offset between client and db time too big. Stopping Suspicious-Replica-Recoverer.')
|
|
602
|
+
return
|
|
603
|
+
|
|
604
|
+
logging.info('Suspicious file replicas recovery starting 1 worker.')
|
|
605
|
+
t = threading.Thread(target=declare_suspicious_replicas_bad,
|
|
606
|
+
kwargs={'once': once,
|
|
607
|
+
'younger_than': younger_than,
|
|
608
|
+
'nattempts': nattempts,
|
|
609
|
+
'vos': vos,
|
|
610
|
+
'limit_suspicious_files_on_rse': limit_suspicious_files_on_rse,
|
|
611
|
+
'json_file_name': json_file_name,
|
|
612
|
+
'sleep_time': sleep_time,
|
|
613
|
+
'active_mode': active_mode})
|
|
614
|
+
t.start()
|
|
615
|
+
logging.info('Waiting for interrupts')
|
|
616
|
+
|
|
617
|
+
# Interruptible joins require a timeout.
|
|
618
|
+
while t.is_alive():
|
|
619
|
+
t.join(timeout=3.14)
|
|
620
|
+
|
|
621
|
+
|
|
622
|
+
def stop(signum: "Optional[int]" = None, frame: "Optional[FrameType]" = None) -> None:
|
|
623
|
+
"""
|
|
624
|
+
Graceful exit.
|
|
625
|
+
"""
|
|
626
|
+
GRACEFUL_STOP.set()
|
|
@@ -0,0 +1,13 @@
|
|
|
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.
|