diff --git a/swh/loader/svn/ra.py b/swh/loader/svn/ra.py --- a/swh/loader/svn/ra.py +++ b/swh/loader/svn/ra.py @@ -150,7 +150,7 @@ elif key == properties.PROP_SPECIAL: # Possibly a symbolic link. We cannot check further at # that moment though, patch(s) not being applied yet - self.link = True + self.link = value is not None elif key == SVN_PROPERTY_EOL: # backup end of line style for file EOL_STYLE[self.fullpath] = value @@ -227,6 +227,10 @@ self.__make_symlink(src) else: # not a real link... self.link = False + elif os.path.islink(self.fullpath): + # path was a symbolic link in previous revision but got the property + # svn:special unset in current one, revert its content to svn link format + self.__make_svnlink() if not is_link: # if a link, do nothing regarding flag if self.executable == EXEC_FLAG: diff --git a/swh/loader/svn/tests/test_loader.py b/swh/loader/svn/tests/test_loader.py --- a/swh/loader/svn/tests/test_loader.py +++ b/swh/loader/svn/tests/test_loader.py @@ -833,8 +833,9 @@ if "properties" in change: for prop, value in change["properties"].items(): file.change_prop(prop, value) - txdelta = file.apply_textdelta() - delta.send_stream(BytesIO(change["data"]), txdelta) + if "data" in change: + txdelta = file.apply_textdelta() + delta.send_stream(BytesIO(change["data"]), txdelta) file.close() root.close() editor.close() @@ -980,3 +981,97 @@ ) == b"../file_with_crlf_eol.txt" ) + + +def test_loader_svn_special_property_unset(swh_storage, tmp_path): + # create a repository + repo_path = os.path.join(tmp_path, "tmprepo") + repos.create(repo_path) + repo_url = f"file://{repo_path}" + + # first commit + add_commit( + repo_url, + ( + "Create a regular file, a link to a file and a link to an " + "external file. Set the svn:special property on the links." + ), + [ + CommitChange( + change_type=CommitChangeType.AddOrUpdate, + path="file.txt", + data=b"Hello world!\n", + ), + CommitChange( + change_type=CommitChangeType.AddOrUpdate, + path="link.txt", + properties={"svn:special": "*"}, + data=b"link ./file.txt", + ), + CommitChange( + change_type=CommitChangeType.AddOrUpdate, + path="external_link.txt", + properties={"svn:special": "*"}, + data=b"link /home/user/data.txt", + ), + ], + ) + + # second commit + add_commit( + repo_url, + "Unset the svn:special property on the links.", + [ + CommitChange( + change_type=CommitChangeType.AddOrUpdate, + path="link.txt", + properties={"svn:special": None}, + ), + CommitChange( + change_type=CommitChangeType.AddOrUpdate, + path="external_link.txt", + properties={"svn:special": None}, + ), + ], + ) + + # instantiate a svn loader checking after each processed revision that + # the repository filesystem it reconstructed does not differ from a subversion + # export of that revision + loader = SvnLoader( + swh_storage, repo_url, destination_path=tmp_path, check_revision=1 + ) + + assert loader.load() == {"status": "eventful"} + assert loader.visit_status() == "full" + + # check loaded objects are those expected + assert get_stats(loader.storage) == { + "content": 5, + "directory": 2, + "origin": 1, + "origin_visit": 1, + "release": 0, + "revision": 2, + "skipped_content": 0, + "snapshot": 1, + } + + root_dir = loader.snapshot.branches[b"HEAD"].target + revision = loader.storage.revision_get([root_dir])[0] + + paths = {} + for entry in loader.storage.directory_ls(revision.directory, recursive=True): + paths[entry["name"]] = entry + + assert paths[b"link.txt"]["perms"] == DentryPerms.content + assert ( + loader.storage.content_get_data(paths[b"link.txt"]["sha1"]) + == b"link ./file.txt" + ) + + assert paths[b"external_link.txt"]["perms"] == DentryPerms.content + assert ( + loader.storage.content_get_data(paths[b"external_link.txt"]["sha1"]) + == b"link /home/user/data.txt" + )