Wiki source code of Filesystem store

Last modified by Thomas Mortagne on 2024/06/24

Hide last authors
Thomas Mortagne 9.1 1 {{box cssClass="floatinginfobox" title="**Contents**"}}{{toc start="2"/}}{{/box}}
Thomas Mortagne 1.68 2
Thomas Mortagne 12.1 3 By default in XWiki the content of attachments and deleted attachments and documents are stored in what we call the "filesystem store". Its default location is ##<permanentdir>/store/file##.
Thomas Mortagne 1.1 4
Thomas Mortagne 10.1 5 It's a simple file based storage in which documents and attachments references are (md5) hashed, to avoid problems with various limited (in terms of encoding and path size) file systems. The filesystem store implements a two stage commit mechanism to maintain integrity even if the database fails to commit the attachment meta-data for example.
Thomas Mortagne 1.1 6
Thomas Mortagne 1.70 7 For example the attachment ##XWikiLogo.png## in document ##Sandbox.WebHome## is stored in the following location: ##/1/0/5d42329a923e687f5dff4887d80098/attachments/9/2/0bc685fa0da28168319c0126def81b##.
Thomas Mortagne 1.1 8
Thomas Mortagne 1.26 9 {{box title="And where is my entity located ?"}}
Thomas Mortagne 1.25 10 {{velocity}}
Thomas Mortagne 1.19 11 {{html}}
12 <form>
Thomas Mortagne 9.1 13 <input type="text" name="reference"#if($request.reference) value="$escapetool.xml($request.reference)#end"/>
Thomas Mortagne 1.20 14 <button name="serialize_document">Document</button>
15 <button name="serialize_attachment">Attachment</button>
Thomas Mortagne 1.19 16 </form>
17 {{/html}}
Thomas Mortagne 1.25 18 {{/velocity}}
Thomas Mortagne 1.19 19
Thomas Mortagne 1.42 20 {{groovy}}
Thomas Mortagne 1.16 21 def hash(str)
22 {
Thomas Mortagne 1.49 23 md5 = org.apache.commons.codec.digest.DigestUtils.md5Hex(str)
Thomas Mortagne 1.71 24 println "1. It's then md5 hashed (##${md5}##)."
Thomas Mortagne 1.32 25
Thomas Mortagne 1.49 26 char0 = md5.charAt(0)
Thomas Mortagne 1.71 27 println "1. A first folder is created with the first character of the hash (##${char0}##)."
Thomas Mortagne 1.32 28
Thomas Mortagne 1.49 29 char1 = md5.charAt(1)
Thomas Mortagne 1.71 30 println "1. A second folder is created with the second character of the hash (##${char1}##)."
Thomas Mortagne 1.32 31
Thomas Mortagne 1.49 32 charn = md5.substring(2)
Thomas Mortagne 1.71 33 println "1. A third folder is created with the remaining characters (##${charn}##)."
Thomas Mortagne 1.9 34
Thomas Mortagne 1.52 35 return "**$char0**/**$char1**/**$charn**"
Thomas Mortagne 1.16 36 }
37
Thomas Mortagne 1.17 38 def encodeDocument(documentReference) {
Thomas Mortagne 1.24 39 path = ""
Thomas Mortagne 1.29 40 if (documentReference.root.type == org.xwiki.model.EntityType.WIKI) {
Thomas Mortagne 1.71 41 println "1. A folder with the wiki name (##${documentReference.root.name}##) is created."
Thomas Mortagne 1.29 42 path = documentReference.root.name
Thomas Mortagne 1.24 43 }
44
Thomas Mortagne 1.49 45 localuid = services.component.getInstance(org.xwiki.model.reference.EntityReferenceSerializer.TYPE_STRING, "local/uid").serialize(documentReference)
Thomas Mortagne 1.71 46 println "1. The document reference is serialized using the local uid serializer (##${localuid}##)."
Thomas Mortagne 1.32 47
Thomas Mortagne 1.24 48 path += "/" + hash(localuid)
49
50 return path;
Thomas Mortagne 1.16 51 }
52
Thomas Mortagne 1.17 53 def encodeAttachment(attachmentReference) {
Thomas Mortagne 1.16 54 path = ""
55 if (reference.parent) {
56 path = encodeDocument(attachmentReference.parent)
57 }
58
Thomas Mortagne 1.49 59 path += "/attachments"
Thomas Mortagne 1.71 60 println "1. An ##attachments## folder is created."
Thomas Mortagne 1.16 61
Thomas Mortagne 1.49 62 name = attachmentReference.name
Thomas Mortagne 1.71 63 println "1. The attachment name (##$name##) is extracted from the reference."
Thomas Mortagne 1.49 64
Thomas Mortagne 1.51 65 path += "/" + hash(name) + ""
Thomas Mortagne 1.16 66 }
67
Thomas Mortagne 1.22 68 if (request.serialize_document != null) {
Thomas Mortagne 1.28 69 reference = services.component.getInstance(org.xwiki.model.reference.EntityReferenceResolver.TYPE_STRING, "relative").resolve(request.reference, org.xwiki.model.EntityType.DOCUMENT)
Thomas Mortagne 1.16 70
Thomas Mortagne 1.70 71 println "{{info}}"
72 path = encodeDocument(reference)
73 println "{{/info}}"
74
75 println ""
76 print "The final path of the folder which contains the content of this attachment is ##$path##."
Thomas Mortagne 1.22 77 } else if (request.serialize_attachment != null) {
Thomas Mortagne 1.23 78 reference = services.component.getInstance(org.xwiki.model.reference.EntityReferenceResolver.TYPE_STRING, "relative").resolve(request.reference, org.xwiki.model.EntityType.ATTACHMENT)
Thomas Mortagne 1.16 79
Thomas Mortagne 1.41 80 println "{{info}}"
Thomas Mortagne 1.36 81 path = encodeAttachment(reference)
Thomas Mortagne 1.41 82 println "{{/info}}"
Thomas Mortagne 1.36 83
Thomas Mortagne 1.33 84 println ""
Thomas Mortagne 1.70 85 print "The final path of the folder which contains the content of this document is ##$path##."
Thomas Mortagne 1.2 86 }
87 {{/groovy}}
Thomas Mortagne 1.18 88 {{/box}}
Thomas Mortagne 1.55 89
Thomas Mortagne 9.1 90 == Attachments and deleted attachments
Thomas Mortagne 1.60 91
Thomas Mortagne 1.65 92 By default, inside the attachment folder, you will find both the current version of the attachment and its history. The file name is always ##f##, it's then optionally followed by the version (when it's a piece of the history) and the original file extension.
Thomas Mortagne 1.55 93
Thomas Mortagne 4.1 94 For example for an image of type png:
95
Thomas Mortagne 1.59 96 * ##f.png##
Thomas Mortagne 1.64 97 * ##fv1.1.png##
98 * ##fv2.1.png##
Thomas Mortagne 1.59 99
Thomas Mortagne 1.55 100 {{version since="16.4.0"}}
Thomas Mortagne 7.1 101 Most of the time, instead of the current version of the attachment, a "link" is created. It's a file which name is suffixed with ##.lnk## which contains the relative path it's representing. In standard condition, it always points to the latest version of the attachment.
Thomas Mortagne 1.59 102
Thomas Mortagne 9.1 103 For example for an image of type png:
Thomas Mortagne 4.1 104
Thomas Mortagne 2.1 105 * ##f.png.lnk## {{info}}contains the string ##fv2.1.png##{{/info}}
Thomas Mortagne 1.63 106 * ##fv1.1.png##
107 * ##fv2.1.png##
Thomas Mortagne 5.1 108 {{/version}}
Thomas Mortagne 1.67 109
Thomas Mortagne 11.2 110 The different between attachments and deleted attachment is the location of that folder inside the document folder:
111 * an attachment is located in ##/attachments/<hash/based/on the attachment name>/##
112 * a deleted is located in ##/deleted-attachments/<hash/based/on the attachment name>/<index of the deleted attachment>/##
Thomas Mortagne 1.67 113
Thomas Mortagne 9.1 114 == Deleted documents
115
Thomas Mortagne 13.1 116 {{todo/}}

Get Connected