diff --git a/CHANGELOG.txt b/CHANGELOG.txt index fab031f389b5a794ec5e103a9342c6fa76a9f60e..6c81cfc0f54bcd26c6a1517150aea449f2d476ae 100644 --- a/CHANGELOG.txt +++ b/CHANGELOG.txt @@ -35,6 +35,8 @@ Drupal x.x.x, xxxx-xx-xx (development version) - PHP Template engine: * add the ability to look for a series of suggested templates. * look for page templates based upon the path. +- content system: + * made it easier for node access modules to work well with each other. Drupal 4.7.0, 2006-05-01 ------------------------ diff --git a/modules/node/node.module b/modules/node/node.module index cc759d5d875b90a7ccd0d8c535b530e921b9aef0..b99ae3ad39851d203358f9493ef799b046d8b6f4 100644 --- a/modules/node/node.module +++ b/modules/node/node.module @@ -488,6 +488,9 @@ function node_save(&$node) { node_invoke_nodeapi($node, 'update'); } + // Update the node access table for this node. + node_access_acquire_grants($node); + // Clear the cache so an anonymous poster can see the node being added or updated. cache_clear_all(); } @@ -916,6 +919,13 @@ function node_menu($may_cache) { 'title' => t("'%name' content type", array('%name' => node_get_name(arg(3)))), 'type' => MENU_CALLBACK); } + + // There is no need to rebuild node_access if there is only 1 record in the table (the default configuration). + if (db_result(db_query('SELECT COUNT(*) FROM {node_access}')) > 1) { + $items[] = array('path' => 'admin/settings/node-access', 'title' => t('node access'), + 'callback' => 'node_access_rebuild_page', + 'access' => user_access('administer nodes')); + } } return $items; @@ -2524,6 +2534,136 @@ function node_db_rewrite_sql($query, $primary_table, $primary_field) { } } +/** + * This function will call module invoke to get a list of grants and then + * write them to the database. It is called at node save, and should be + * called by modules whenever something other than a node_save causes + * the permissions on a node to change. + * + * This function is the only function that should write to the node_access + * table. + * + * @param $node + * The $node to acquire grants for. + */ +function node_access_acquire_grants($node) { + $grants = module_invoke_all('node_access_records', $node); + if (!$grants) { + $grants[] = array('realm' => 'all', 'gid' => 0, 'grant_view' => 1, 'grant_update' => 0, 'grant_delete' => 0); + } + else { + // retain grants by highest priority + $grant_by_priority = array(); + foreach ($grants as $g) { + $grant_by_priority[intval($g['priority'])][] = $g; + } + krsort($grant_by_priority); + $grants = array_shift($grant_by_priority); + } + + node_access_write_grants($node, $grants); +} + +/** + * This function will write a list of grants to the database, deleting + * any pre-existing grants. If a realm is provided, it will only + * delete grants from that realm, but it will always delete a grant + * from the 'all' realm. Modules which utilize node_access can + * use this function when doing mass updates due to widespread permission + * changes. + * + * @param $node + * The $node being written to. All that is necessary is that it contain a nid. + * @param $grants + * A list of grants to write. Each grant is an array that must contain the + * following keys: realm, gid, grant_view, grant_update, grant_delete. + * The realm is specified by a particular module; the gid is as well, and + * is a module-defined id to define grant privileges. each grant_* field + * is a boolean value. + * @param $realm + * If provided, only read/write grants for that realm. + * @param $delete + * If false, do not delete records. This is only for optimization purposes, + * and assumes the caller has already performed a mass delete of some form. + */ +function node_access_write_grants($node, $grants, $realm = NULL, $delete = TRUE) { + if ($delete) { + $query = 'DELETE FROM {node_access} WHERE nid = %d'; + if ($realm) { + $query .= " AND realm in ('%s', 'all')"; + } + db_query($query, $node->nid, $realm); + } + + // only perform work when node_access modules are active + if (count(module_implements('node_grants'))) { + // This optimization reduces the number of db inserts a little bit. We could + // optimize further for mass updates if we wanted. + $values = array(); + $query = ''; + foreach ($grants as $grant) { + if ($realm && $realm != $grant['realm']) { + continue; + } + // Only write grants; denies are implicit. + if ($grant['grant_view'] || $grant['grant_update'] || $grant['grant_delete']) { + $query .= ($query ? ', ' : '') . "(%d, '%s', %d, %d, %d, %d)"; + + $values[] = $node->nid; + $values[] = $grant['realm']; + $values[] = $grant['gid']; + $values[] = $grant['grant_view']; + $values[] = $grant['grant_update']; + $values[] = $grant['grant_delete']; + } + } + + if ($values) { + $query = "INSERT INTO {node_access} (nid, realm, gid, grant_view, grant_update, grant_delete) VALUES " . $query; + db_query($query, $values); + } + } +} + +function node_access_rebuild_page() { + $form['markup'] = array( + '#prefix' => '<p>', + '#value' => t('Rebuilding the node_access table is necessary immediately after uninstalling a module that utilizes the node_access system. Each node will have its access control recalculated. This may take a while if your site has many nodes.'), + '#suffix' => '</p>', + ); + $form['submit'] = array( + '#type' => 'submit', + '#value' => t('Rebuild node access'), + ); + return drupal_get_form('node_access_rebuild', $form); +} + +/** + * rebuild the node access database + */ +function node_access_rebuild_submit() { + node_access_rebuild(); + drupal_set_message(t('The node access table has been rebuilt.')); +} + +function node_access_rebuild() { + db_query("DELETE FROM {node_access}"); + // only recalculate if site is using a node_access module + if (count(module_implements('node_grants'))) { + // If not in 'safe mode', increase the maximum execution time: + if (!ini_get('safe_mode')) { + set_time_limit(240); + } + $result = db_query("SELECT nid FROM {node}"); + while ($node = db_fetch_object($result)) { + node_access_acquire_grants(node_load($node->nid)); + } + } + else { + // not using any node_access modules. add the default grant. + db_query("INSERT INTO {node_access} VALUES (0, 0, 'all', 1, 0, 0)"); + } +} /** * @} End of "defgroup node_access". */