$guid), $to_be_serialised); // Sanity check if ((!is_array($to_be_serialised)) || (count($to_be_serialised)==0)) throw new ExportException("No such entity GUID:$guid"); // Now serialise the result to XML $wrapper = new stdClass; // Construct header $wrapper->header = new stdClass; $wrapper->header->date = date("r"); $wrapper->header->timestamp = time(); $wrapper->header->domain = $CONFIG->wwwroot; $wrapper->header->exported_by = guid_to_uuid($_SESSION['id']); // Construct data $wrapper->data = $to_be_serialised; return serialise_object_to_xml($wrapper, "elggexport"); } /** * XML 2 Array function. * Taken from http://www.bytemycode.com/snippets/snippet/445/ * @license UNKNOWN - Please contact if you are the original author of this code. * @author UNKNOWN */ function __xml2array($xml) { $xmlary = array(); $reels = '/<(\w+)\s*([^\/>]*)\s*(?:\/>|>(.*)<\/\s*\\1\s*>)/s'; $reattrs = '/(\w+)=(?:"|\')([^"\']*)(:?"|\')/'; preg_match_all($reels, $xml, $elements); foreach ($elements[1] as $ie => $xx) { $xmlary[$ie]["name"] = $elements[1][$ie]; if ($attributes = trim($elements[2][$ie])) { preg_match_all($reattrs, $attributes, $att); foreach ($att[1] as $ia => $xx) $xmlary[$ie]["attributes"][$att[1][$ia]] = $att[2][$ia]; } $cdend = strpos($elements[3][$ie], "<"); if ($cdend > 0) { $xmlary[$ie]["text"] = substr($elements[3][$ie], 0, $cdend - 1); } if (preg_match($reels, $elements[3][$ie])) $xmlary[$ie]["elements"] = __xml2array($elements[3][$ie]); else if ($elements[3][$ie]) { $xmlary[$ie]["text"] = $elements[3][$ie]; } } return $xmlary; } $IMPORTED_DATA = array(); $IMPORTED_OBJECT_COUNTER = 0; /** * This function processes an element, passing elements to the plugin stack to see if someone will * process it. * If nobody processes the top level element, the sub level elements are processed. */ function __process_element(array $dom) { global $IMPORTED_DATA, $IMPORTED_OBJECT_COUNTER; foreach ($dom as $element) { // See if anyone handles this element, return true if it is. $handled = trigger_plugin_hook("import", "all", array("name" => $element['name'], "element" => $element), $to_be_serialised); // If not, then see if any of its sub elements are handled if (!$handled) { if (isset($element['elements'])) __process_element($element['elements']); } else { $IMPORTED_OBJECT_COUNTER ++; // Increment validation counter $IMPORTED_DATA[] = $handled; // Return the constructed object } } } /** * Import an XML serialisation of an object. * This will make a best attempt at importing a given xml doc. * * @param string $xml * @return array An array of imported objects (these have already been saved). * @throws Exception if there was a problem importing the data. */ function import($xml) { global $IMPORTED_DATA, $IMPORTED_OBJECT_COUNTER; $IMPORTED_DATA = array(); $IMPORTED_OBJECT_COUNTER = 0; $IMPORTED_GUID_MAP = array(); $dom = __xml2array($xml); __process_element($dom); if ($IMPORTED_OBJECT_COUNTER!= count($IMPORTED_DATA)) throw new ImportException("Not all elements were imported."); return $IMPORTED_DATA; } /** * Generate a UUID from a given GUID. * * @param int $guid The GUID of an object. */ function guid_to_uuid($guid) { global $CONFIG; return "UUID:".md5($CONFIG->wwwroot) . ":$guid"; } /** * Test to see if a given uuid is for this domain, returning true if so. * @param $uuid * @return bool */ function is_uuid_this_domain($uuid) { global $CONFIG; $domain = md5($CONFIG->wwwroot); $tmp = explode(":",$uuid); if (strcmp($tmp[1], $domain) == 0) return true; return false; } /** * This function attempts to retrieve a previously imported entity via its UUID. * * @param $uuid */ function get_entity_from_uuid($uuid) { $uuid = sanitise_string($uuid); $entities = get_entities_from_metadata("import_uuid", $uuid); if ($entities) return $entities[0]; return false; } /** * Tag a previously created guid with the uuid it was imported on. * * @param int $guid * @param string $uuid */ function add_uuid_to_guid($guid, $uuid) { $guid = (int)$guid; $uuid = sanitise_string($uuid); return create_metadata($guid, "import_uuid", $uuid); } /** * This function serialises an object recursively into an XML representation. * The function attempts to call $data->export() which expects a stdClass in return, otherwise it will attempt to * get the object variables using get_object_vars (which will only return public variables!) * @param $data object The object to serialise. * @param $n int Level, only used for recursion. * @return string The serialised XML output. */ function serialise_object_to_xml($data, $name = "", $n = 0) { $classname = ($name=="" ? get_class($data) : $name); $vars = method_exists($data, "export") ? get_object_vars($data->export()) : get_object_vars($data); $output = ""; if (($n==0) || ( is_object($data) && !($data instanceof stdClass))) $output = "<$classname>"; foreach ($vars as $key => $value) { $output .= "<$key type=\"".gettype($value)."\">"; if (is_object($value)) $output .= serialise_object_to_xml($value, $key, $n+1); else if (is_array($value)) $output .= serialise_array_to_xml($value, $n+1); else $output .= htmlentities($value); $output .= "\n"; } if (($n==0) || ( is_object($data) && !($data instanceof stdClass))) $output .= "\n"; return $output; } /** * Serialise an array. * * @param array $data * @param int $n Used for recursion * @return string */ function serialise_array_to_xml(array $data, $n = 0) { $output = ""; if ($n==0) $output = "\n"; foreach ($data as $key => $value) { $item = "array_item"; if (is_numeric($key)) $output .= "<$item name=\"$key\" type=\"".gettype($value)."\">"; else { $item = $key; $output .= "<$item type=\"".gettype($value)."\">"; } if (is_object($value)) $output .= serialise_object_to_xml($value, "", $n+1); else if (is_array($value)) $output .= serialise_array_to_xml($value, $n+1); else $output .= htmlentities($value); $output .= "\n"; } if ($n==0) $output = "\n"; return $output; } /** * Export exception * * @package Elgg * @subpackage Exceptions * */ class ExportException extends Exception {} /** * Import exception * * @package Elgg * @subpackage Exceptions */ class ImportException extends Exception {} ?>