diff --git a/README_SETUP.md b/README_SETUP.md
index 374ea306100aff4cb4f6cbb37f4e9f1eee6f7abc..dc667da8aa5877132c1212d2ddd2827e85992118 100644
--- a/README_SETUP.md
+++ b/README_SETUP.md
@@ -103,7 +103,10 @@ like this, check out the "Authentication" section in the [configuration document
 Now would be a good time to continue with the [tutorials](tutorials/index).
 
 ## Run Unit Tests
-tox
+
+- Run all tests: `tox` or `make unittest`
+- Run a specific test file: e.g. `tox -- unittests/test_schema.py`
+- Run a specific test function: e.g. `tox -- unittests/test_schema.py::test_config_files`
 
 ## Documentation ##
 
diff --git a/src/caosdb/apiutils.py b/src/caosdb/apiutils.py
index 12b2b32060aeece374f560f7be4fe7a741c07db6..a376068c372c1b6f460c7927467b8da8df328545 100644
--- a/src/caosdb/apiutils.py
+++ b/src/caosdb/apiutils.py
@@ -39,7 +39,9 @@ from caosdb.common.datatype import (BOOLEAN, DATETIME, DOUBLE, FILE, INTEGER,
                                     REFERENCE, TEXT, is_reference)
 from caosdb.common.models import (Container, Entity, File, Property, Query,
                                   Record, RecordType, execute_query,
-                                  get_config)
+                                  get_config, SPECIAL_ATTRIBUTES)
+
+import logging
 
 
 def new_record(record_type, name=None, description=None,
@@ -186,10 +188,6 @@ def getCommitIn(folder):
         return t.readline().strip()
 
 
-COMPARED = ["name", "role", "datatype", "description",
-            "id", "path", "checksum", "size"]
-
-
 def compare_entities(old_entity: Entity, new_entity: Entity):
     """
     Compare two entites.
@@ -213,7 +211,7 @@ def compare_entities(old_entity: Entity, new_entity: Entity):
     if old_entity is new_entity:
         return (olddiff, newdiff)
 
-    for attr in COMPARED:
+    for attr in SPECIAL_ATTRIBUTES:
         try:
             oldattr = old_entity.__getattribute__(attr)
             old_entity_attr_exists = True
@@ -302,51 +300,25 @@ def compare_entities(old_entity: Entity, new_entity: Entity):
     return (olddiff, newdiff)
 
 
-def copy_entity(entity: Entity):
-    """
-    Return a copy of entity.
-
-    If deep == True return a deep copy, recursively copying all sub entities.
+def merge_entities(entity_a: Entity, entity_b: Entity):
     """
-    print(entity)
-    new: Optional[Entity] = None
-    if entity.role == "File":
-        new = File()
-    elif entity.role == "Property":
-        new = Property()
-    elif entity.role == "RecordType":
-        new = RecordType()
-    elif entity.role == "Record":
-        new = Record()
-    elif entity.role == "Entity":
-        new = Entity()
-    else:
-        raise RuntimeError("Unkonwn role.")
-
-    # Copy special attributes:
-    # TODO: this might rise an exception when copying
-    #       special file attributes like checksum and size.
-    for attribute in COMPARED + ["value"]:
-        val = getattr(entity, attribute)
-        if val is not None:
-            setattr(new, attribute, val)
-
-    # Copy parents:
-    for p in entity.parents:
-        new.add_parent(p)
+    Merge entity_b into entity_a such that they have the same parents and properties.
 
-    # Copy properties:
-    for p in entity.properties:
-        new.add_property(p, importance=entity.get_importance(p))
+    datatype, unit, value, name and description will only be changed in entity_a if they
+    are None for entity_a and set for entity_b. If there is a corresponding value
+    for entity_a different from None a RuntimeError will be raised informing of an
+    unresolvable merge conflict.
 
-    return new
+    The merge operation is done in place.
 
+    Returns entity_a.
 
-def merge_entities(entity_a: Entity, entity_b: Entity):
-    """
-    Merge entity_b into entity_a such that they have the same parents and properties.
+    WARNING: This function is currently experimental and insufficiently tested. Use with care.
     """
 
+    logging.warning(
+        "This function is currently experimental and insufficiently tested. Use with care.")
+
     # Compare both entities:
     diff_r1, diff_r2 = compare_entities(entity_a, entity_b)
 
@@ -372,9 +344,17 @@ def merge_entities(entity_a: Entity, entity_b: Entity):
                 else:
                     raise RuntimeError("Merge conflict.")
         else:
-            entity_a.add_property(
-                entity_b.get_property(key),
-                importance=entity_b.get_importance(key))
+            # TODO: This is a temporary FIX for
+            #       https://gitlab.indiscale.com/caosdb/src/caosdb-pylib/-/issues/105
+            entity_a.add_property(id=entity_b.get_property(key).id,
+                                  name=entity_b.get_property(key).name,
+                                  datatype=entity_b.get_property(key).datatype,
+                                  value=entity_b.get_property(key).value,
+                                  unit=entity_b.get_property(key).unit,
+                                  importance=entity_b.get_importance(key))
+            # entity_a.add_property(
+            #     entity_b.get_property(key),
+            #     importance=entity_b.get_importance(key))
 
     for special_attribute in ("name", "description"):
         sa_a = getattr(entity_a, special_attribute)
diff --git a/src/caosdb/common/models.py b/src/caosdb/common/models.py
index 181750aae6fd3e1aeab2c61b59f53d8b8111d5bd..6475bc99ec825e102d5eac1b38d506247c11ebcb 100644
--- a/src/caosdb/common/models.py
+++ b/src/caosdb/common/models.py
@@ -79,6 +79,10 @@ ALL = "ALL"
 NONE = "NONE"
 
 
+SPECIAL_ATTRIBUTES = ["name", "role", "datatype", "description",
+                      "id", "path", "checksum", "size"]
+
+
 class Entity(object):
 
     """Entity is a generic CaosDB object.
@@ -121,6 +125,48 @@ class Entity(object):
         self.id = id
         self.state = None
 
+
+    def copy(self):
+        """
+        Return a copy of entity.
+
+        If deep == True return a deep copy, recursively copying all sub entities.
+
+        Standard properties are copied using add_property.
+        Special attributes, as defined by the global variable SPECIAL_ATTRIBUTES and additionaly
+        the "value" are copied using setattr.
+        """
+        if self.role == "File":
+            new = File()
+        elif self.role == "Property":
+            new = Property()
+        elif self.role == "RecordType":
+            new = RecordType()
+        elif self.role == "Record":
+            new = Record()
+        elif self.role == "Entity":
+            new = Entity()
+        else:
+            raise RuntimeError("Unkonwn role.")
+
+        # Copy special attributes:
+        # TODO: this might rise an exception when copying
+        #       special file attributes like checksum and size.
+        for attribute in SPECIAL_ATTRIBUTES + ["value"]:
+            val = getattr(self, attribute)
+            if val is not None:
+                setattr(new, attribute, val)
+
+        # Copy parents:
+        for p in self.parents:
+            new.add_parent(p)
+
+        # Copy properties:
+        for p in self.properties:
+            new.add_property(p, importance=self.get_importance(p))
+
+        return new
+
     @property
     def version(self):
         if self._version is not None or self._wrapped_entity is None:
diff --git a/unittests/test_apiutils.py b/unittests/test_apiutils.py
index fcedc380d15c06acaf93122922e513d025a1563e..2859a0ec529d5b1a5f51b57b1407320ba98f8d73 100644
--- a/unittests/test_apiutils.py
+++ b/unittests/test_apiutils.py
@@ -29,7 +29,9 @@
 import caosdb as db
 import caosdb.apiutils
 from caosdb.apiutils import (apply_to_ids, compare_entities, create_id_query,
-                             resolve_reference, copy_entity, merge_entities)
+                             resolve_reference, merge_entities)
+
+from caosdb.common.models import SPECIAL_ATTRIBUTES
 
 
 def test_apply_to_ids():
@@ -221,15 +223,83 @@ def test_copy_entities():
     r.add_property(name="D", value=[3, 4, 7], importance="OBLIGATORY")
     r.description = "A fancy test record"
 
-    c = copy_entity(r)
+    c = r.copy()
 
-    assert c != r
+    assert c is not r
     assert c.name == "A"
+    assert c.role == r.role
     assert c.parents[0].name == "B"
-    # Currently parents and properties are always individual to a copy:
-    assert c.parents[0] != r.parents[0]
+    # parent and property objects are not shared among copy and original:
+    assert c.parents[0] is not r.parents[0]
 
     for i in [0, 1]:
-        assert c.properties[i] != r.properties[i]
-        assert c.properties[i].value == r.properties[i].value
+        assert c.properties[i] is not r.properties[i]
+        for special in SPECIAL_ATTRIBUTES:
+            assert getattr(c.properties[i], special) == getattr(r.properties[i], special)
         assert c.get_importance(c.properties[i]) == r.get_importance(r.properties[i])
+
+
+def test_merge_entities():
+    r = db.Record(name="A")
+    r.add_parent(name="B")
+    r.add_property(name="C", value=4, importance="OBLIGATORY")
+    r.add_property(name="D", value=[3, 4, 7], importance="OBLIGATORY")
+    r.description = "A fancy test record"
+
+    r2 = db.Record()
+    r2.add_property(name="F", value="text")
+    merge_entities(r2, r)
+    assert r2.get_parents()[0].name == "B"
+    assert r2.get_property("C").name == "C"
+    assert r2.get_property("C").value == 4
+    assert r2.get_property("D").name == "D"
+    assert r2.get_property("D").value == [3, 4, 7]
+
+    assert r2.get_property("F").name == "F"
+    assert r2.get_property("F").value == "text"
+
+
+def test_merge_bug_109():
+    rt = db.RecordType(name="TestBug")
+    p = db.Property(name="test_bug_property", datatype=db.LIST(db.INTEGER))
+
+    r_b = db.Record(name="TestRecord")
+    r_b.add_parent(rt)
+    r_b.add_property(p, value=[18, 19])
+
+    r_a = db.Record(name="TestRecord")
+    r_a.add_parent(rt)
+
+    merge_entities(r_a, r_b)
+
+    assert r_b.get_property("test_bug_property").value == [18, 19]
+    assert r_a.get_property("test_bug_property").value == [18, 19]
+
+    assert "<Value>18</Value>\n    <Value>19</Value>" in str(r_b)
+    assert "<Value>18</Value>\n    <Value>19</Value>\n    <Value>18</Value>\n    <Value>19</Value>" not in str(r_b)
+
+    assert "<Value>18</Value>\n    <Value>19</Value>" in str(r_a)
+    assert "<Value>18</Value>\n    <Value>19</Value>\n    <Value>18</Value>\n    <Value>19</Value>" not in str(r_a)
+
+
+@pytest.mark.xfail
+def test_bug_109():
+    rt = db.RecordType(name="TestBug")
+    p = db.Property(name="test_bug_property", datatype=db.LIST(db.INTEGER))
+
+    r_b = db.Record(name="TestRecord")
+    r_b.add_parent(rt)
+    r_b.add_property(p, value=[18, 19])
+
+    r_a = db.Record(name="TestRecord")
+    r_a.add_parent(rt)
+    r_a.add_property(r_b.get_property("test_bug_property"))
+
+    assert r_b.get_property("test_bug_property").value == [18, 19]
+    assert r_a.get_property("test_bug_property").value == [18, 19]
+
+    assert "<Value>18</Value>\n    <Value>19</Value>" in str(r_b)
+    assert "<Value>18</Value>\n    <Value>19</Value>\n    <Value>18</Value>\n    <Value>19</Value>" not in str(r_b)
+
+    assert "<Value>18</Value>\n    <Value>19</Value>" in str(r_a)
+    assert "<Value>18</Value>\n    <Value>19</Value>\n    <Value>18</Value>\n    <Value>19</Value>" not in str(r_a)