-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathNodeAssociationToModelServer.kt
64 lines (54 loc) · 2.59 KB
/
NodeAssociationToModelServer.kt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
package org.modelix.model.sync.bulk
import org.modelix.model.ModelIndex
import org.modelix.model.api.IBranch
import org.modelix.model.api.IMutableModel
import org.modelix.model.api.IReadableNode
import org.modelix.model.api.IWritableNode
import org.modelix.model.api.NodeReference
import org.modelix.model.api.PNodeAdapter
import org.modelix.model.api.getOriginalOrCurrentReference
import org.modelix.model.api.getOriginalReference
import org.modelix.model.area.PArea
import org.modelix.model.data.NodeData
class NodeAssociationToModelServer(val branch: IBranch) : INodeAssociation {
private val modelIndex
get() = ModelIndex.get(branch.transaction, NodeData.Companion.ID_PROPERTY_KEY)
override fun resolveTarget(sourceNode: IReadableNode): IWritableNode? {
val ref = sourceNode.getOriginalOrCurrentReference()
return modelIndex.find(ref).map { PNodeAdapter(it, branch) }.firstOrNull()?.asWritableNode()
?: PArea(branch).resolveNode(NodeReference(ref))?.asWritableNode()
}
override fun associate(sourceNode: IReadableNode, targetNode: IWritableNode) {
val expected = sourceNode.getOriginalOrCurrentReference()
if (expected != targetNode.getOriginalOrCurrentReference()) {
targetNode.setPropertyValue(NodeData.ID_PROPERTY_REF, expected)
}
}
}
class NodeAssociationFromModelServer(val branch: IBranch, val targetModel: IMutableModel) : INodeAssociation {
private val pendingAssociations = HashMap<Long, String>()
private fun nodeId(sourceNode: IReadableNode): Long {
val pnode = sourceNode.asLegacyNode() as PNodeAdapter
require(pnode.branch == branch) {
"Node is from a different branch. [node = $sourceNode, expected: $branch, actual: ${pnode.branch}]"
}
return pnode.nodeId
}
override fun resolveTarget(sourceNode: IReadableNode): IWritableNode? {
return (pendingAssociations[nodeId(sourceNode)] ?: sourceNode.getOriginalReference())
?.let { targetModel.resolveNode(NodeReference(it)) }
}
override fun associate(sourceNode: IReadableNode, targetNode: IWritableNode) {
val id = nodeId(sourceNode)
val expected = sourceNode.getOriginalOrCurrentReference()
if (expected != targetNode.getOriginalOrCurrentReference()) {
pendingAssociations[id] = targetNode.getNodeReference().serialize()
}
}
fun writeAssociations() {
for (entry in pendingAssociations) {
branch.writeTransaction.setProperty(entry.key, NodeData.ID_PROPERTY_KEY, entry.value)
}
pendingAssociations.clear()
}
}