<?php /** * @file * A list of orphaned Islandora objects. */ /** * Builds the Orphaned Islandora Objects management form. * * @param array $form * An array representing a form within Drupal. * @param array $form_state * An array containing the Drupal form state. * * @return array * An array containing the form to be rendered. */ function islandora_manage_orphaned_objects_form(array $form, array $form_state) { if (isset($form_state['show_confirm'])) { $pids = $form_state['pids_to_delete']; $form['confirm_message'] = array( '#type' => 'item', '#markup' => format_plural(count($form_state['pids_to_delete']), 'Are you sure you want to delete this object? This action cannot be reversed.', 'Are you sure you want to delete these @count objects? This action cannot be reversed.'), ); if (count($pids) <= 10) { $form['pids_to_delete'] = array( '#type' => 'markup', '#theme' => 'item_list', '#list_type' => 'ol', ); $options = array('attributes' => array('target' => '_blank')); foreach ($pids as $pid) { $form['pids_to_delete']['#items'][] = l($pid, "/islandora/object/{$pid}", $options); } } $form['confirm_submit'] = array( '#type' => 'submit', '#value' => t('Confirm'), '#weight' => 2, '#submit' => array('islandora_manage_orphaned_objects_confirm_submit'), ); $form['cancel_submit'] = array( '#type' => 'submit', '#value' => t('Cancel'), '#weight' => 3, ); } else { drupal_set_message(t('This page lists objects that have at least one parent, according to their RELS-EXT, that does not exist in the Fedora repository. These orphans might exist due to a failed batch ingest, their parents being deleted, or a variety of other reasons. Some of these orphans may exist intentionally. Please be cautious when deleting, as this action is irreversible.'), 'warning'); $orphaned_objects = islandora_get_orphaned_objects(); $query_method = variable_get('islandora_orphaned_objects_backend', 'SPARQL'); module_load_include('inc', 'islandora', 'includes/utilities'); $rows = array(); foreach ($orphaned_objects as $orphaned_object) { if ($query_method == 'SPARQL') { $pid = $orphaned_object['object']['value']; $title = $orphaned_object['title']['value']; } elseif ($query_method == 'Solr') { $pid = $orphaned_object['PID']; $title = $orphaned_object['object_label']; } if (islandora_namespace_accessible($pid)) { $rows[$pid] = array(l($title . " (" . $pid . ")", "islandora/object/$pid")); } } ksort($rows); $form['management_table'] = array( '#type' => 'tableselect', '#header' => array(t('Object')), '#options' => $rows, '#attributes' => array(), '#empty' => t('No orphaned objects were found.'), ); if (!empty($rows)) { $form['submit_selected'] = array( '#type' => 'submit', '#name' => 'islandora-orphaned-objects-submit-selected', '#validate' => array('islandora_delete_selected_orphaned_objects_validate'), '#submit' => array('islandora_delete_orphaned_objects_submit'), '#value' => t('Delete Selected'), ); $form['submit_all'] = array( '#type' => 'submit', '#name' => 'islandora-orphaned-objects-submit-all', '#submit' => array('islandora_delete_orphaned_objects_submit'), '#value' => t('Delete All'), ); } } return $form; } /** * Validation for the Islandora Orphaned Objects management form. * * @param array $form * An array representing a form within Drupal. * @param array $form_state * An array containing the Drupal form state. */ function islandora_delete_selected_orphaned_objects_validate(array $form, array $form_state) { $selected = array_filter($form_state['values']['management_table']); if (empty($selected)) { form_error($form['management_table'], t('At least one object must be selected to delete!')); } } /** * Submit handler for the delete buttons in the workflow management form. * * @param array $form * An array representing a form within Drupal. * @param array $form_state * An array containing the Drupal form state. */ function islandora_delete_orphaned_objects_submit(array &$form, array &$form_state) { if ($form_state['triggering_element']['#name'] == 'islandora-orphaned-objects-submit-selected') { $selected = array_keys(array_filter($form_state['values']['management_table'])); } else { $selected = array_keys($form_state['values']['management_table']); } $form_state['pids_to_delete'] = $selected; // Rebuild to show the confirm form. $form_state['rebuild'] = TRUE; $form_state['show_confirm'] = TRUE; } /** * Submit handler for the workflow management confirm form. * * @param array $form * An array representing a form within Drupal. * @param array $form_state * An array containing the Drupal form state. */ function islandora_manage_orphaned_objects_confirm_submit(array $form, array &$form_state) { $batch = islandora_delete_orphaned_objects_create_batch($form_state['pids_to_delete']); batch_set($batch); } /** * Query for orphaned objects. * * @return array * An array containing the results of the orphaned objects queries. */ function islandora_get_orphaned_objects() { $query_method = variable_get('islandora_orphaned_objects_backend', 'SPARQL'); if ($query_method == 'Solr') { // Solr query for all objects. $collection_field = variable_get('islandora_solr_member_of_collection_field', 'RELS_EXT_isMemberOfCollection_uri_ms'); $label_field = variable_get('islandora_solr_object_label_field', 'fgs_label_s'); $member_field = variable_get('islandora_solr_member_of_field', 'RELS_EXT_isMemberOf_uri_ms'); $params = "PID, " . $label_field . ", " . $collection_field . ", " . $member_field; $query = "PID:*"; $qp = new islandoraSolrQueryProcessor(); $qp->buildQuery($query); $qp->solrParams['fl'] = $params; $qp->solrLimit = 1000000000; // Check islandora_compound_object filters, include compound children if necessary. if (variable_get('islandora_compound_object_hide_child_objects_solr', TRUE)) { $fq = variable_get('islandora_compound_object_solr_fq', '-RELS_EXT_isConstituentOf_uri_mt:[* TO *]'); if (!empty($fq)) { // Delete islandora_compound_object_solr_fq from the list of filters. $filters = $qp->solrParams['fq']; if (($key = array_search($fq, $filters)) !== FALSE) { unset($filters[$key]); $qp->solrParams['fq'] = $filters; } } } $qp->executeQuery(FALSE); try { $results = $qp->islandoraSolrResult['response']['objects']; } catch (Exception $e) { watchdog_exception('Islandora', $e, 'Got an exception searching for parent objects .', array(), WATCHDOG_ERROR); $results = array(); } $orphaned_objects = array(); $already_checked = array(); $missing_parents = array(); // Check all results for PIDs that don't exist. foreach ($results AS $result) { if (array_key_exists($collection_field, $result['solr_doc'])) { foreach ($result['solr_doc'][$collection_field] AS $collection) { if (in_array($collection, $missing_parents)) { $orphaned_objects[] = $result; } elseif (!in_array($collection, $already_checked)) { $test = islandora_identify_missing_parents($collection); if (!$test) { $orphaned_objects[] = $result; $missing_parents[] = $collection; } $already_checked[] = $collection; } } } if (array_key_exists($member_field, $result['solr_doc'])) { foreach ($result['solr_doc'][$member_field] AS $membership) { if (in_array($membership, $missing_parents)) { $orphaned_objects[] = $result; } elseif (!in_array($membership, $already_checked)) { $test = islandora_identify_missing_parents($membership); if (!$test) { $orphaned_objects[] = $result; $missing_parents[] = $membership; } $already_checked[] = $membership; } } } } $results = $orphaned_objects; } elseif ($query_method == "SPARQL") { $connection = islandora_get_tuque_connection(); // SPARQL: get orphaned objects, exclude any with a living parent. $object_query = <<<EOQ !prefix SELECT DISTINCT ?object ?title WHERE { ?object <fedora-model:hasModel> <info:fedora/fedora-system:FedoraObject-3.0> ; ?p ?otherobject . ?object <fedora-model:label> ?title; OPTIONAL { ?otherobject <fedora-model:hasModel> ?model . } . FILTER (!bound(?model)) # Filter by "parent" relationships FILTER (!dead_parent_relationships) # Exclude objects with live parents OPTIONAL { !live_parent_relationships . ?liveparent <fedora-model:hasModel> <info:fedora/fedora-system:FedoraObject-3.0> . } !optionals !filters FILTER (!bound(?liveparent)) } ORDER BY ?object EOQ; $parent_relationships = module_invoke_all('islandora_solution_pack_child_relationships', 'all'); $parent_relationships['prefix'] = array_unique($parent_relationships['prefix']); $parent_relationships['predicate'] = array_unique($parent_relationships['predicate']); if (count($parent_relationships['predicate']) == 0) { // No predicates to search for. Exit early. return array(); } $optionals = (array) module_invoke('islandora_xacml_api', 'islandora_basic_collection_get_query_optionals', 'view'); $filter_modules = array( 'islandora_xacml_api', 'islandora', ); $filters = array(); foreach ($filter_modules as $module) { $filters = array_merge_recursive($filters, (array) module_invoke($module, 'islandora_basic_collection_get_query_filters', 'view')); } $filter_map = function ($filter) { return "FILTER($filter)"; }; $parent_map = function ($parent) { return "?object $parent ?liveparent"; }; // Use separate queries for different object types. $sparql_query_objects = format_string($object_query, array( '!optionals' => !empty($optionals) ? ('OPTIONAL {{' . implode('} UNION {', $optionals) . '}}') : '', '!filters' => !empty($filters) ? implode(' ', array_map($filter_map, $filters)) : '', '!dead_parent_relationships' => '?p = ' . implode(' || ?p = ', $parent_relationships['predicate']), '!live_parent_relationships' => '{' . implode(' } UNION { ', array_map($parent_map, $parent_relationships['predicate'])) . '}', '!prefix' => implode("\n", $parent_relationships['prefix']), )); $results = $connection->repository->ri->sparqlQuery($sparql_query_objects); } return $results; } /** * Constructs the batch that will go out and delete objects. * * @param array $pids * The array of pids to be deleted. * * @return array * An array detailing the batch that is about to be run. */ function islandora_delete_orphaned_objects_create_batch(array $pids) { // Set up a batch operation. $batch = array( 'operations' => array( array('islandora_delete_orphaned_objects_batch_operation', array($pids)), ), 'title' => t('Deleting the selected objects...'), 'init_message' => t('Preparing to delete objects.'), 'progress_message' => t('Time elapsed: @elapsed <br/>Estimated time remaining @estimate.'), 'error_message' => t('An error has occurred.'), 'finished' => 'islandora_delete_orphaned_objects_batch_finished', 'file' => drupal_get_path('module', 'islandora') . '/includes/orphaned_objects.inc', ); return $batch; } /** * Solr query to check for deceased parents. * */ function islandora_identify_missing_parents($parent) { $parent_params = "PID"; $parent_test = substr($parent, strpos($parent, '/') +1); $parent_query = 'PID:"' . $parent_test . '"'; $qp = new islandoraSolrQueryProcessor(); $qp->buildQuery($parent_query); $qp->solrParams['fl'] = $parent_params; $qp->solrLimit = 1000000000; $qp->executeQuery(FALSE); try { $parent_results = $qp->islandoraSolrResult['response']['objects']; } catch (Exception $e) { watchdog_exception('Islandora', $e, 'Got an exception searching for parent objects .', array(), WATCHDOG_ERROR); $parent_results = array(); } return($parent_results); } /** * Constructs and performs the deleting batch operation. * * @param array $pids * An array of pids to be deleted. * @param array $context * The context of the Drupal batch. */ function islandora_delete_orphaned_objects_batch_operation(array $pids, array &$context) { if (empty($context['sandbox'])) { $context['sandbox'] = array(); $context['sandbox']['progress'] = 0; $context['sandbox']['pids'] = $pids; $context['sandbox']['total'] = count($pids); $context['results']['success'] = array(); } if (!empty($context['sandbox']['pids'])) { $target_pid = array_pop($context['sandbox']['pids']); $target_object = islandora_object_load($target_pid); $context['message'] = t('Deleting @label (@pid) (@current of @total)...', array( '@label' => $target_object->label, '@pid' => $target_pid, '@current' => $context['sandbox']['progress'], '@total' => $context['sandbox']['total'], )); islandora_delete_object($target_object); $object_check = islandora_object_load($target_pid); if ($object_check) { drupal_set_message(t('Could not delete %pid. You may not have permission to manage this object.', array( '%pid' => $target_pid, )), 'error'); } else { $context['results']['success'][] = $target_pid; } $context['sandbox']['progress']++; } $context['finished'] = ($context['sandbox']['total'] == 0) ? 1 : ($context['sandbox']['progress'] / $context['sandbox']['total']); } /** * Finished function for the orphaned objects delete batch. * * @param bool $success * Whether the batch was successful or not. * @param array $results * An array containing the results of the batch operations. * @param array $operations * The operations array that was used in the batch. */ function islandora_delete_orphaned_objects_batch_finished($success, array $results, array $operations) { if ($success) { $message = format_plural(count($results['success']), 'One object deleted.', '@count objects deleted.'); } else { $message = t('Finished with an error.'); } drupal_set_message($message); }