void SyncManagerImpl::AddCreateOpForElementRecurs(RemoteSyncPeer& peer, const ElementPtr& element)
{
	// Construct a CreateOperation for this element
	XGuid parentGuid = kInvalidXGuid;
	ElementPtr parent = element->GetParent();
	if (parent)
	{
		parentGuid = parent->GetGUID();
	}

	{
		CreateOperationPtr createOp = new CreateOperation(
			element->GetElementType(),
			element->GetName(),
			element->GetGUID(),
			parentGuid,
			element->GetXValue(),
			m_syncContext->GetAuthorityLevel(),
			m_syncContext);

		// Add to the list of outgoing ops
		peer.SendOp(createOp);
	}

	// If this is an object element...
	ObjectElementPtr objElement = ObjectElement::Cast(element);
	if (objElement)
	{
		// Recurs to the children of this element
		for (int32 i = 0; i < objElement->GetElementCount(); ++i)
		{
			AddCreateOpForElementRecurs(peer, objElement->GetElementAt(i));
		}
	}
	else
	{
		// If the element is an array, add Insert operations for all its entries
		ArrayElement* arrayElement = reflection_cast<ArrayElement>(element);
		if (arrayElement != nullptr)
		{
			int32 numEntries = arrayElement->GetCount();
			for (int32 i = 0; i < numEntries; ++i)
			{
				InsertOperationPtr insertOp = new InsertOperation(
					element->GetGUID(),
					i,
					arrayElement->GetXValue(i),
					m_syncContext->GetAuthorityLevel(),
					m_syncContext);

				// Add to the list of outgoing ops
				peer.SendOp(insertOp);
			}
		}
	}
}
void SyncManagerImpl::DeleteDataFromUserRecurs(const ElementPtr& element, const UserDiscriminator& discriminator)
{
	ObjectElementPtr objectElement = ObjectElement::Cast(element);
	if (objectElement)
	{
		if (discriminator(objectElement->GetOwnerID()))
		{
#if defined(SYNC_DEBUG)
			LogInfo("%s Deleting object %s with owner %i", m_syncContext->GetLocalUser()->GetName().GetString().c_str(), objectElement->GetName().GetString().c_str(), objectElement->GetOwnerID());
#endif
			// Create a delete operation to represent this change
			DeleteOperationPtr deleteOp = new DeleteOperation(element->GetGUID(), element->GetParent()->GetGUID(), m_syncContext);

			// Execute the operation
			deleteOp->Apply(m_syncContext);

			// Add the op to the list of ops applied sync the last sync, so that remote machines 
			// are sent the delete operation
			m_syncContext->AddAppliedOperation(deleteOp);

			// Add the op to the list of pending notifications, to notify the user on this machine that the element has been deleted
			m_pendingNotifications.push(deleteOp);
		}
		else
		{
			// Recurs to the children of this element
			for (int32 i = objectElement->GetElementCount() - 1; i >= 0; --i)
			{
				DeleteDataFromUserRecurs(objectElement->GetElementAt(i), discriminator);
			}
		}
	}
}
	void SyncTestbed::CompareElementRecurs(const ElementPtr& element1, const ElementPtr& element2) const
	{
		Assert::AreEqual(element1->GetName()->GetString(), element2->GetName()->GetString());
		Assert::IsTrue(element1->GetGUID() == element2->GetGUID());
		Assert::IsTrue(element1->GetElementType() == element2->GetElementType());
		Assert::IsTrue(element1->GetXValue() == element2->GetXValue());

		if (element1->GetElementType() == ElementType::ObjectType)
		{
			ObjectElementPtr objElement1 = ObjectElement::Cast(element1);
			Assert::IsNotNull(objElement1.get());
			ObjectElementPtr objElement2 = ObjectElement::Cast(element2);
			Assert::IsNotNull(objElement2.get());

			Assert::AreEqual(objElement1->GetElementCount(), objElement2->GetElementCount());

			for (int32 i = 0; i < objElement1->GetElementCount(); ++i)
			{
				ElementPtr childElement1 = objElement1->GetElementAt(i);

				ElementPtr childElement2 = objElement2->GetElement(childElement1->GetName());
				Assert::IsNotNull(childElement2.get());

				CompareElementRecurs(childElement1, childElement2);
			}
		}
		else if (IsFrom<ArrayElement>(element1))
		{
			ArrayElement* arrayElement1 = reflection_cast<ArrayElement>(element1);
			Assert::IsNotNull(arrayElement1);

			ArrayElement* arrayElement2 = reflection_cast<ArrayElement>(element2);
			Assert::IsNotNull(arrayElement2);

			Assert::AreEqual(arrayElement1->GetCount(), arrayElement2->GetCount());

			for (int32 i = 0; i < arrayElement1->GetCount(); ++i)
			{
				Assert::IsTrue(arrayElement1->GetXValue(i) == arrayElement2->GetXValue(i));
			}
		}
	}