Skip to main content
summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorEike Stepper2013-10-01 06:54:49 +0000
committerEike Stepper2013-10-01 06:54:49 +0000
commit6d6126fb998a4cfb7cdd20dcbdf9ac202d5845a5 (patch)
treec5a79a9a6208ae6cf4f89454db2a8da6213ca6e3
parent48c07bf1f2acc05313f6f4b3bfaee8db208b1d66 (diff)
downloadcdo-6d6126fb998a4cfb7cdd20dcbdf9ac202d5845a5.tar.gz
cdo-6d6126fb998a4cfb7cdd20dcbdf9ac202d5845a5.tar.xz
cdo-6d6126fb998a4cfb7cdd20dcbdf9ac202d5845a5.zip
[418393] [Security] CDOResourceFolder.getNodes() fails for
CDOPermission.NONE https://bugs.eclipse.org/bugs/show_bug.cgi?id=418393
-rw-r--r--plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/BaseCDORevision.java288
-rw-r--r--plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/DelegatingCDORevision.java16
-rw-r--r--plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/InternalCDORevision.java10
-rw-r--r--plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/StubCDORevision.java16
-rw-r--r--plugins/org.eclipse.emf.cdo/src/org/eclipse/emf/internal/cdo/view/AbstractCDOView.java40
5 files changed, 236 insertions, 134 deletions
diff --git a/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/BaseCDORevision.java b/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/BaseCDORevision.java
index c3500d8569..c6c07e2c29 100644
--- a/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/BaseCDORevision.java
+++ b/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/BaseCDORevision.java
@@ -75,6 +75,8 @@ public abstract class BaseCDORevision extends AbstractCDORevision
private static final int RESOURCE_NODE_NAME_INDEX = 1;
+ private static final int RESOURCE_FOLDER_NODES_INDEX = 2;
+
private static final byte UNSET_OPCODE = 0;
private static final byte SET_NULL_OPCODE = 1;
@@ -166,8 +168,17 @@ public abstract class BaseCDORevision extends AbstractCDORevision
{
clearValues();
- String name = in.readString();
- doSetValue(RESOURCE_NODE_NAME_INDEX, name);
+ EClass eClass = getEClass();
+ EStructuralFeature[] features = getAllPersistentFeatures();
+ readValue(in, eClass, features, RESOURCE_NODE_NAME_INDEX, true);
+
+ if (getClassInfo().isResourceFolder())
+ {
+ if (!readValue(in, eClass, features, RESOURCE_FOLDER_NODES_INDEX, true))
+ {
+ flags &= ~UNCHUNKED_FLAG;
+ }
+ }
}
}
else
@@ -217,6 +228,71 @@ public abstract class BaseCDORevision extends AbstractCDORevision
}
/**
+ * @since 4.3
+ */
+ public boolean readValues(CDODataInput in) throws IOException
+ {
+ EClass owner = getEClass();
+ EStructuralFeature[] features = getAllPersistentFeatures();
+ initValues(features);
+
+ boolean unchunked = true;
+ for (int i = 0; i < features.length; i++)
+ {
+ unchunked = readValue(in, owner, features, i, unchunked);
+ }
+
+ return unchunked;
+ }
+
+ private boolean readValue(CDODataInput in, EClass owner, EStructuralFeature[] features, int i, boolean unchunked)
+ throws IOException
+ {
+ Object value;
+ byte unsetState = in.readByte();
+ switch (unsetState)
+ {
+ case UNSET_OPCODE:
+ return unchunked;
+
+ case SET_NULL_OPCODE:
+ setValue(i, CDORevisionData.NIL);
+ return unchunked;
+ }
+
+ EStructuralFeature feature = features[i];
+ if (feature.isMany())
+ {
+ CDOList list = in.readCDOList(owner, feature);
+ if (unchunked)
+ {
+ int size = list.size();
+ if (size != 0)
+ {
+ Object lastElement = list.get(size - 1);
+ if (lastElement == InternalCDOList.UNINITIALIZED || lastElement instanceof CDOElementProxy)
+ {
+ unchunked = false;
+ }
+ }
+ }
+
+ value = list;
+ }
+ else
+ {
+ value = in.readCDOFeatureValue(feature);
+ if (TRACER.isEnabled())
+ {
+ TRACER.format("Read feature {0}: {1}", feature.getName(), value);
+ }
+ }
+
+ setValue(i, value);
+ return unchunked;
+ }
+
+ /**
* @since 4.0
*/
public void write(CDODataOutput out, int referenceChunk) throws IOException
@@ -244,8 +320,14 @@ public abstract class BaseCDORevision extends AbstractCDORevision
{
if (getClassInfo().isResourceNode())
{
- String name = getResourceNodeName();
- out.writeString(name);
+ EClass eClass = getEClass();
+ EStructuralFeature[] features = getAllPersistentFeatures();
+ writeValue(out, eClass, features, RESOURCE_NODE_NAME_INDEX, referenceChunk);
+
+ if (getClassInfo().isResourceFolder())
+ {
+ writeValue(out, eClass, features, RESOURCE_FOLDER_NODES_INDEX, referenceChunk);
+ }
}
}
else
@@ -299,6 +381,63 @@ public abstract class BaseCDORevision extends AbstractCDORevision
}
/**
+ * @since 4.3
+ */
+ public void writeValues(CDODataOutput out, int referenceChunk) throws IOException
+ {
+ EClass owner = getEClass();
+ EStructuralFeature[] features = getAllPersistentFeatures();
+ for (int i = 0; i < features.length; i++)
+ {
+ writeValue(out, owner, features, i, referenceChunk);
+ }
+ }
+
+ private void writeValue(CDODataOutput out, EClass owner, EStructuralFeature[] features, int i, int referenceChunk)
+ throws IOException
+ {
+ EStructuralFeature feature = features[i];
+ Object value = getValue(i);
+ if (value == null)
+ {
+ // Feature is NOT set
+ out.writeByte(UNSET_OPCODE);
+ return;
+ }
+
+ // Feature IS set
+ if (value == CDORevisionData.NIL)
+ {
+ // Feature IS null
+ out.writeByte(SET_NULL_OPCODE);
+ return;
+ }
+
+ // Feature is NOT null
+ out.writeByte(SET_NOT_NULL_OPCODE);
+ if (feature.isMany())
+ {
+ CDOList list = (CDOList)value;
+ out.writeCDOList(owner, feature, list, referenceChunk);
+ }
+ else
+ {
+ checkNoFeatureMap(feature);
+ if (feature instanceof EReference)
+ {
+ value = out.getIDProvider().provideCDOID(value);
+ }
+
+ if (TRACER.isEnabled())
+ {
+ TRACER.format("Writing feature {0}: {1}", feature.getName(), value);
+ }
+
+ out.writeCDOFeatureValue(feature, value);
+ }
+ }
+
+ /**
* @see #write(CDODataOutput, int)
* @since 3.0
*/
@@ -722,22 +861,14 @@ public abstract class BaseCDORevision extends AbstractCDORevision
synchronized (this)
{
- CDOPermission permission = getPermission();
- if (permission != CDOPermission.WRITE)
- {
- setPermission(CDOPermission.WRITE);
- }
-
try
{
+ bypassPermissionChecks(true);
setValue(featureIndex, list);
}
finally
{
- if (permission != CDOPermission.WRITE)
- {
- setPermission(permission);
- }
+ bypassPermissionChecks(false);
}
}
}
@@ -860,7 +991,7 @@ public abstract class BaseCDORevision extends AbstractCDORevision
protected void setValue(int featureIndex, Object value)
{
- checkFrozen(featureIndex, value);
+ checkUnfrozen(featureIndex, value);
checkWritable();
doSetValue(featureIndex, value);
}
@@ -882,24 +1013,30 @@ public abstract class BaseCDORevision extends AbstractCDORevision
return (CDOList)getValue(i);
}
- private void checkFrozen(int featureIndex, Object value)
+ private void checkUnfrozen(int featureIndex, Object value)
{
if ((flags & FROZEN_FLAG) != 0)
{
+ // Exception 1: LoadPermissionsRequest needs to "reload" revision values in case the original permission was NONE.
+ // In this case BYPASS_PERMISSION_CHECKS_FLAG is set.
+ if ((flags & BYPASS_PERMISSION_CHECKS_FLAG) != 0)
+ {
+ return;
+ }
+
Object oldValue = getValue(featureIndex);
- // Exception 1: Setting an empty list as the value for an isMany feature, is
- // allowed if the old value is null. This is a case of lazy initialization.
+ // Exception 2: Setting an empty list as the value for an isMany feature, is allowed if the old value is null.
+ // This is a case of lazy initialization.
boolean newIsEmptyList = value instanceof EList<?> && ((EList<?>)value).size() == 0;
if (newIsEmptyList && oldValue == null)
{
return;
}
- // Exception 2a: Replacing a temp ID with a regular ID is allowed (happens during
- // postCommit of new objects)
- // Exception 2b: Replacing a temp ID with another temp ID is also allowed (happens
- // when changes are imported in a PushTx).
+ // Exception 3a: Replacing a temp ID with a regular ID is allowed (happens during postCommit of new objects)
+ // Exception 3b: Replacing a temp ID with another temp ID is also allowed (happens when changes are imported in a
+ // PushTx).
if (oldValue instanceof CDOIDTemp && value instanceof CDOID)
{
return;
@@ -940,109 +1077,6 @@ public abstract class BaseCDORevision extends AbstractCDORevision
}
}
- private void writeValues(CDODataOutput out, int referenceChunk) throws IOException
- {
- EClass owner = getEClass();
- EStructuralFeature[] features = getAllPersistentFeatures();
- for (int i = 0; i < features.length; i++)
- {
- EStructuralFeature feature = features[i];
- Object value = getValue(i);
- if (value == null)
- {
- // Feature is NOT set
- out.writeByte(UNSET_OPCODE);
- continue;
- }
-
- // Feature IS set
- if (value == CDORevisionData.NIL)
- {
- // Feature IS null
- out.writeByte(SET_NULL_OPCODE);
- continue;
- }
-
- // Feature is NOT null
- out.writeByte(SET_NOT_NULL_OPCODE);
- if (feature.isMany())
- {
- CDOList list = (CDOList)value;
- out.writeCDOList(owner, feature, list, referenceChunk);
- }
- else
- {
- checkNoFeatureMap(feature);
- if (feature instanceof EReference)
- {
- value = out.getIDProvider().provideCDOID(value);
- }
-
- if (TRACER.isEnabled())
- {
- TRACER.format("Writing feature {0}: {1}", feature.getName(), value);
- }
-
- out.writeCDOFeatureValue(feature, value);
- }
- }
- }
-
- private boolean readValues(CDODataInput in) throws IOException
- {
- EClass owner = getEClass();
- EStructuralFeature[] features = getAllPersistentFeatures();
- initValues(features);
-
- boolean unchunked = true;
- for (int i = 0; i < features.length; i++)
- {
- Object value;
- EStructuralFeature feature = features[i];
- byte unsetState = in.readByte();
- switch (unsetState)
- {
- case UNSET_OPCODE:
- continue;
-
- case SET_NULL_OPCODE:
- setValue(i, CDORevisionData.NIL);
- continue;
- }
-
- if (feature.isMany())
- {
- CDOList list = in.readCDOList(owner, feature);
- if (unchunked)
- {
- int size = list.size();
- if (size != 0)
- {
- Object lastElement = list.get(size - 1);
- if (lastElement == InternalCDOList.UNINITIALIZED || lastElement instanceof CDOElementProxy)
- {
- unchunked = false;
- }
- }
- }
-
- value = list;
- }
- else
- {
- value = in.readCDOFeatureValue(feature);
- if (TRACER.isEnabled())
- {
- TRACER.format("Read feature {0}: {1}", feature.getName(), value);
- }
- }
-
- setValue(i, value);
- }
-
- return unchunked;
- }
-
public static void checkNoFeatureMap(EStructuralFeature feature)
{
if (FeatureMapUtil.isFeatureMap(feature))
@@ -1102,7 +1136,7 @@ public abstract class BaseCDORevision extends AbstractCDORevision
builder.append("FROZEN");
}
- if ((flags & CDOPermission.READ.getBits()) != 0)
+ if ((flags & READ_PERMISSION_FLAG) != 0)
{
if (builder.length() != 0)
{
@@ -1112,7 +1146,7 @@ public abstract class BaseCDORevision extends AbstractCDORevision
builder.append("READ");
}
- if ((flags & CDOPermission.WRITE.getBits()) != 0)
+ if ((flags & WRITE_PERMISSION_FLAG) != 0)
{
if (builder.length() != 0)
{
diff --git a/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/DelegatingCDORevision.java b/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/DelegatingCDORevision.java
index bf93c5d84c..95e319b9f2 100644
--- a/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/DelegatingCDORevision.java
+++ b/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/DelegatingCDORevision.java
@@ -333,6 +333,14 @@ public abstract class DelegatingCDORevision implements InternalCDORevision
getDelegate().read(in);
}
+ /**
+ * @since 4.3
+ */
+ public boolean readValues(CDODataInput in) throws IOException
+ {
+ return getDelegate().readValues(in);
+ }
+
public void write(CDODataOutput out, int referenceChunk) throws IOException
{
getDelegate().write(out, referenceChunk);
@@ -346,6 +354,14 @@ public abstract class DelegatingCDORevision implements InternalCDORevision
getDelegate().write(out, referenceChunk, securityContext);
}
+ /**
+ * @since 4.3
+ */
+ public void writeValues(CDODataOutput out, int referenceChunk) throws IOException
+ {
+ getDelegate().writeValues(out, referenceChunk);
+ }
+
public void convertEObjects(CDOIDProvider oidProvider)
{
getDelegate().convertEObjects(oidProvider);
diff --git a/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/InternalCDORevision.java b/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/InternalCDORevision.java
index 26ec0f652e..87187ee650 100644
--- a/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/InternalCDORevision.java
+++ b/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/InternalCDORevision.java
@@ -116,6 +116,11 @@ public interface InternalCDORevision extends CDORevision, CDORevisionData, CDORe
public void read(CDODataInput in) throws IOException;
/**
+ * @since 4.3
+ */
+ public boolean readValues(CDODataInput in) throws IOException;
+
+ /**
* @since 3.0
*/
public void write(CDODataOutput out, int referenceChunk) throws IOException;
@@ -126,6 +131,11 @@ public interface InternalCDORevision extends CDORevision, CDORevisionData, CDORe
public void write(CDODataOutput out, int referenceChunk, CDOBranchPoint securityContext) throws IOException;
/**
+ * @since 4.3
+ */
+ public void writeValues(CDODataOutput out, int referenceChunk) throws IOException;
+
+ /**
* @since 3.0
*/
public void convertEObjects(CDOIDProvider oidProvider);
diff --git a/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/StubCDORevision.java b/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/StubCDORevision.java
index dd9b7994ac..a95435cca5 100644
--- a/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/StubCDORevision.java
+++ b/plugins/org.eclipse.emf.cdo.common/src/org/eclipse/emf/cdo/spi/common/revision/StubCDORevision.java
@@ -148,6 +148,14 @@ public class StubCDORevision extends AbstractCDORevision
throw new UnsupportedOperationException(getExceptionMessage());
}
+ /**
+ * @since 4.3
+ */
+ public boolean readValues(CDODataInput in) throws IOException
+ {
+ throw new UnsupportedOperationException(getExceptionMessage());
+ }
+
public void write(CDODataOutput out, int referenceChunk) throws IOException
{
throw new UnsupportedOperationException(getExceptionMessage());
@@ -161,6 +169,14 @@ public class StubCDORevision extends AbstractCDORevision
throw new UnsupportedOperationException(getExceptionMessage());
}
+ /**
+ * @since 4.3
+ */
+ public void writeValues(CDODataOutput out, int referenceChunk) throws IOException
+ {
+ throw new UnsupportedOperationException(getExceptionMessage());
+ }
+
public void convertEObjects(CDOIDProvider oidProvider)
{
throw new UnsupportedOperationException(getExceptionMessage());
diff --git a/plugins/org.eclipse.emf.cdo/src/org/eclipse/emf/internal/cdo/view/AbstractCDOView.java b/plugins/org.eclipse.emf.cdo/src/org/eclipse/emf/internal/cdo/view/AbstractCDOView.java
index 7f265d9dd0..30e942c006 100644
--- a/plugins/org.eclipse.emf.cdo/src/org/eclipse/emf/internal/cdo/view/AbstractCDOView.java
+++ b/plugins/org.eclipse.emf.cdo/src/org/eclipse/emf/internal/cdo/view/AbstractCDOView.java
@@ -29,6 +29,7 @@ import org.eclipse.emf.cdo.common.model.CDOClassifierRef;
import org.eclipse.emf.cdo.common.model.CDOModelUtil;
import org.eclipse.emf.cdo.common.protocol.CDOProtocolConstants;
import org.eclipse.emf.cdo.common.revision.CDOIDAndVersion;
+import org.eclipse.emf.cdo.common.revision.CDOList;
import org.eclipse.emf.cdo.common.revision.CDORevision;
import org.eclipse.emf.cdo.common.revision.CDORevisionData;
import org.eclipse.emf.cdo.common.revision.CDORevisionKey;
@@ -641,16 +642,32 @@ public abstract class AbstractCDOView extends CDOCommitHistoryProviderImpl<CDOOb
EReference nodesFeature = EresourcePackage.eINSTANCE.getCDOResourceFolder_Nodes();
EAttribute nameFeature = EresourcePackage.eINSTANCE.getCDOResourceNode_Name();
- int size = folderRevision.data().size(nodesFeature);
+
+ CDOList list;
+
+ try
+ {
+ folderRevision.bypassPermissionChecks(true);
+ list = folderRevision.getList(nodesFeature);
+ }
+ finally
+ {
+ folderRevision.bypassPermissionChecks(false);
+ }
+
+ CDOStore store = getStore();
+ int size = list.size();
for (int i = 0; i < size; i++)
{
- Object value = folderRevision.data().get(nodesFeature, i);
- value = getStore().resolveProxy(folderRevision, nodesFeature, i, value);
+ Object value = list.get(i);
+ value = store.resolveProxy(folderRevision, nodesFeature, i, value);
- CDORevision childRevision = getLocalRevision((CDOID)convertObjectToID(value));
- if (name.equals(childRevision.data().get(nameFeature, 0)))
+ CDOID childID = (CDOID)convertObjectToID(value);
+ InternalCDORevision childRevision = getLocalRevision(childID);
+ String childName = (String)childRevision.get(nameFeature, 0);
+ if (name.equals(childName))
{
- return childRevision.getID();
+ return childID;
}
}
@@ -1387,7 +1404,16 @@ public abstract class AbstractCDOView extends CDOCommitHistoryProviderImpl<CDOOb
try
{
- CDOID id = isRoot ? getSession().getRepositoryInfo().getRootResourceID() : getResourceNodeID(path);
+ CDOID id;
+ if (isRoot)
+ {
+ id = getSession().getRepositoryInfo().getRootResourceID();
+ }
+ else
+ {
+ id = getResourceNodeID(path);
+ }
+
resource.cdoInternalSetID(id);
registerObject(resource);
if (isRoot)

Back to the top