Wiki source code of Saving Data

Version 7.1 by Thomas Mortagne on 2017/03/24

Hide last authors
Vincent Massol 1.1 1 {{box cssClass="floatinginfobox" title="**Contents**"}}
2 {{toc/}}
3 {{/box}}
4
5 There are times when you wish to save data. The XWiki platform offers various places that you can use depending on your needs. Here's a rundown of all the options you have.
6
7 = Transient Saves =
8
9 == Execution Context ==
10
Manuel Smeria 6.5 11 If the data to be saved should only be saved for the duration of a request (i.e. a call to an XWiki URL) then you should declare a property in the **Execution Context**.
Vincent Massol 1.1 12
13 For example, from Java in a Component:
14
15 {{code language="java"}}
16 import org.xwiki.execution.*;
17
18 @Inject Execution execution;
19 ...
Andreas Jonsson 6.1 20 // Declaring
Vincent Massol 1.1 21 ExecutionContext ec = execution.getContext();
Andreas Jonsson 6.1 22 ec.newProperty("mykey").initial(initialValue).declare();
Vincent Massol 1.1 23 ...
Andreas Jonsson 6.1 24 // Updating
25 ec.setProperty("mykey", newValue);
26 ...
Vincent Massol 1.1 27 // Retrieving
28 Object myvalue = e.getProperty("mykey");
29 {{/code}}
30
Andreas Jonsson 6.4 31 === Execution Context Property Declaration ===
32
Manuel Smeria 6.5 33 The declaration is currently optional, the property will be implicitly declared when set the first time. The declaration can be made in an ##ExecutionContextInitializer## component. The initialize method of all such components that are available in the classpath and have been specified in a ##components.txt## file will be called when the execution context is first initialized during the request.
Andreas Jonsson 6.1 34
35 {{code language="java"}}
36 import org.xwiki.context.ExecutionContext;
37 import org.xwiki.context.ExecutionContextInitializer;
38 import org.xwiki.context.ExecutionContextException;
39 import org.xwiki.component.annotation.Component;
40
41 @Component
42 public class DeclareMyPropertyExecutionContextInitializer implements ExecutionContextInitializer
43 {
44 @Override
45 public void initialize(ExecutionContext executionContext)
46 {
Andreas Jonsson 6.2 47 if (!executionContext.hasProperty("mykey")) {
48 executionContext.newProperty("mykey").type(String.class).nonNull().initial("").declare();
Andreas Jonsson 6.1 49 }
50 }
51 }
52 {{/code}}
53
54 In the above example you can see that it is possible to set various attributes on the property when it is beeing declared. The attributes currently supported are listed in the table below.
55
56 During a request cycle, some components may activate a //clean// or a //cloned// execution context by pushing to the execution context stack. You can control how you want your property to be managed in these situations by setting attributes.
57
Andreas Jonsson 6.3 58 |=(% style="width:25%" %)Attribute |=Parameters |=Description
Manuel Smeria 6.5 59 |cloneValue()| - |Indicate that you want the value to be cloned when the execution context is cloned or when your property is inherited from a parent execution context
60 |makeFinal()| - |Indicate that an exception should be thrown if someone tries to replace the initial value object
61 |inherited()| - |Indicate that the property should be inherited by any new execution context that is pushed within the current request
62 |nonNull()| - |Indicate that an exception should be thrown if ##null## is set as the property value
Andreas Jonsson 6.3 63 |type(Class<?> type)|(((
Andreas Jonsson 6.1 64 ; type
65 : the class that the value object should be typechecked against
Andreas Jonsson 6.3 66 )))|Set a type for performing typechecking when updating the value object.
Andreas Jonsson 6.1 67
Andreas Jonsson 6.4 68 === Wiki Page Access to Execution Context ===
69
Manuel Smeria 6.5 70 {{info}}
Thomas Mortagne 7.1 71 We currently [[cannot access easily the Execution Context from a wiki page>>https://jira.xwiki.org/browse/XCOMMONS-242]] but this should be improved. Currently it's not available from Velocity, but it is from Groovy.
Manuel Smeria 6.5 72 {{/info}}
Vincent Massol 1.1 73
Manuel Smeria 6.5 74 From a wiki page:
75
Vincent Massol 1.1 76 {{code}}
77 {{groovy}}
78 import org.xwiki.context.*
79
80 def ec = services.component.getInstance(Execution.class).getContext()
81 ec.setProperty("mykey", myvalue)
82 {{/groovy}}
83 {{/code}}
84
Manuel Smeria 6.5 85 {{info}}
86 From a wiki page it's much easier to use the older **XWiki Context** (which is supposed to be fully replaced by the Execution Context at some point in the future).
87 {{/info}}
Vincent Massol 1.1 88
Manuel Smeria 6.5 89 For example:
90
Vincent Massol 1.1 91 {{code}}
92 {{velocity}}
93 $xcontext.put("mykey", value)
94 {{/velocity}}
95 {{/code}}
96
Manuel Smeria 6.5 97 {{info}}
98 It's also possible to save data in the HTTP Request even though it's recommended to use the Execution Context instead (since the Execution Context is independent of the execution environment and will work everywhere: Servlets, Portlets, JavaSE, etc).
99 {{/info}}
Vincent Massol 4.1 100
Manuel Smeria 6.5 101 For example from Velocity in a wiki page:
102
Vincent Massol 4.1 103 {{code}}
104 {{velocity}}
105 $request.setAttribute("mykey", value)
106 {{/velocity}}
107 {{/code}}
108
Vincent Massol 1.1 109 == Servlet Session ==
110
Vincent Massol 4.1 111 If the data should last a little longer (i.e. for example span more than 1 request), you could save it in the Servlet Session (note that in this case your code will only work in a Servlet Environment).
Vincent Massol 1.1 112
113 For example, from Java in a Component:
114
115 {{code language="java"}}
116 import org.xwiki.container.*;
117
118 @Inject Container container;
119 ...
120 Request request = container.getRequest();
121 // Note that this is a bit of a hack and the notion of Session exists in the Container class (getSession()) but the Session interface is empty at the moment, making it useless)
122 if (request instanceof ServletRequest) {
123 HttpServletRequest servletRequest = (ServletRequest) request;
124 HttpSession session = servletRequest.getSession();
125 session.setAttribute("mykey", myvalue);
126 }
127 {{/code}}
128
Vincent Massol 5.1 129 Example from a wiki page:
130
131 {{code}}
132 {{velocity}}
133 $request.getSession().setAttribute("mykey", myvalue)
134 {{/velocity}}
135 {{/code}}
136
Vincent Massol 1.1 137 == Temporary Directory ==
138
Manuel Smeria 6.5 139 If the data should last even a little longer than a session, you could save it in a file in the environment's temporary directory.
Vincent Massol 1.1 140
141 For example, from Java in a Component:
142
143 {{code language="java"}}
144 import org.xwiki.environment.*;
145 import org.apache.commons.io.*;
146
147 @Inject Environment environment;
148 ...
149 File tmpDir = environment.getTemporaryDirectory();
150 FileUtils.write(tmpDir, "something");
151 {{/code}}
152
153 == Servlet Context ==
154
Vincent Massol 4.1 155 If the Data should last as long as the web application is up, you could use the Servlet Context. It's not very easy to access it but it's possible (note that in this case your code will only work in a Servlet Environment).
Vincent Massol 1.1 156
157 For example, from Java in a Component:
158
159 {{code language="java"}}
160 import org.xwiki.container.*;
161 import com.xpn.xwiki.*;
162 import com.xpn.xwiki.web.*;
163
164 @Inject Execution execution;
165 ...
166 // Get the older XWiki Context
167 XWikiContext xc = (XWikiContext) execution.getContext().getProperty("xwikicontext");
168 XWikiEngineContext ec = xc.getEngineContext();
169 ec.setAttribute("mykey", myvalue);
170 {{/code}}
171
172 = Permanent Saves =
173
174 == Wiki Model ==
175
Manuel Smeria 6.5 176 The best place to save data is usually in the wiki itself (See the [[Data Model>>DevGuide.DataModel]] and the [[API Guide>>DevGuide.APIGuide]] for some examples). There are several places where you can store data in the model:
Andreas Jonsson 6.1 177
Manuel Smeria 6.5 178 * In the document content itself
179 * In an XObject attached to a document
180 * In an attachment
Vincent Massol 1.1 181
182 == Permanent Directory ==
183
Manuel Smeria 6.5 184 Another possibility is to use the environment's permanent directory.
Vincent Massol 1.1 185
186 For example, from Java in a Component:
187
188 {{code language="java"}}
189 import org.xwiki.environment.*;
190 import org.apache.commons.io.*;
191
192 @Inject Environment environment;
193 ...
194 File permDir = environment.getPermanentDirectory();
195 FileUtils.write(permDir, "something");
196 {{/code}}

Get Connected