Classes in this File | Line Coverage | Branch Coverage | Complexity | ||||
Forward |
|
| 1.56;1.56 | ||||
Forward$1 |
|
| 1.56;1.56 | ||||
Forward$ForwardRouting |
|
| 1.56;1.56 |
1 | /* | |
2 | * Copyright 2004-2008 the Seasar Foundation and the Others. | |
3 | * | |
4 | * Licensed under the Apache License, Version 2.0 (the "License"); | |
5 | * you may not use this file except in compliance with the License. | |
6 | * You may obtain a copy of the License at | |
7 | * | |
8 | * http://www.apache.org/licenses/LICENSE-2.0 | |
9 | * | |
10 | * Unless required by applicable law or agreed to in writing, software | |
11 | * distributed under the License is distributed on an "AS IS" BASIS, | |
12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, | |
13 | * either express or implied. See the License for the specific language | |
14 | * governing permissions and limitations under the License. | |
15 | */ | |
16 | package org.seasar.cubby.action; | |
17 | ||
18 | import static org.seasar.cubby.CubbyConstants.ATTR_ROUTINGS; | |
19 | ||
20 | import java.io.IOException; | |
21 | import java.lang.reflect.Method; | |
22 | import java.util.Collections; | |
23 | import java.util.HashMap; | |
24 | import java.util.List; | |
25 | import java.util.Map; | |
26 | import java.util.regex.Pattern; | |
27 | ||
28 | import javax.servlet.RequestDispatcher; | |
29 | import javax.servlet.ServletException; | |
30 | import javax.servlet.http.HttpServletRequest; | |
31 | import javax.servlet.http.HttpServletResponse; | |
32 | ||
33 | import org.seasar.cubby.routing.PathResolver; | |
34 | import org.seasar.cubby.routing.Routing; | |
35 | import org.seasar.cubby.util.CubbyUtils; | |
36 | import org.seasar.cubby.util.QueryStringBuilder; | |
37 | import org.seasar.framework.container.S2Container; | |
38 | import org.seasar.framework.container.factory.SingletonS2ContainerFactory; | |
39 | import org.seasar.framework.log.Logger; | |
40 | import org.seasar.framework.util.ClassUtil; | |
41 | import org.seasar.framework.util.StringUtil; | |
42 | ||
43 | /** | |
44 | * 指定されたパスにフォワードする {@link ActionResult} です。 | |
45 | * <p> | |
46 | * アクションメソッドの戻り値としてこのインスタンスを指定することで、指定されたパスにフォワードします。 | |
47 | * </p> | |
48 | * <p> | |
49 | * 使用例1 : フォワード先を相対パスで指定 | |
50 | * | |
51 | * <pre> | |
52 | * return new Forward("list.jsp"); | |
53 | * </pre> | |
54 | * | |
55 | * </p> | |
56 | * <p> | |
57 | * 使用例2 : フォワード先を絶対パスで指定 | |
58 | * | |
59 | * <pre> | |
60 | * return new Forward("/todo/list.jsp"); | |
61 | * </pre> | |
62 | * | |
63 | * </p> | |
64 | * <p> | |
65 | * 使用例2 : フォワード先をクラスとメソッド名で指定 | |
66 | * | |
67 | * <pre> | |
68 | * return new Forward(TodoListAction.class, "show"); | |
69 | * </pre> | |
70 | * | |
71 | * <p> | |
72 | * 使用例3 : フォワード先をクラスとメソッド名で指定(paramメソッドによるパラメータつき) | |
73 | * | |
74 | * <pre> | |
75 | * return new Forward(TodoListAction.class, "show").param("value1", "12345"); | |
76 | * </pre> | |
77 | * | |
78 | * </p> | |
79 | * <p> | |
80 | * 使用例3 : フォワード先をクラスとメソッド名で指定(Mapによるパラメータつき) | |
81 | * | |
82 | * <pre> | |
83 | * Map<String, String[]> parameters = new HashMap(); | |
84 | * parameters.put("value1", new String[] { "12345" }); | |
85 | * return new Forward(TodoListAction.class, "show", parameters); | |
86 | * </pre> | |
87 | * | |
88 | * </p> | |
89 | * <p> | |
90 | * フォワード前には {@link Action#invokePreRenderMethod(Method)} を実行します。 フォワード後には | |
91 | * {@link Action#invokePostRenderMethod(Method)} を実行し、フラッシュメッセージをクリアします。 | |
92 | * </p> | |
93 | * | |
94 | * @author baba | |
95 | * @since 1.0.0 | |
96 | */ | |
97 | public class Forward implements ActionResult { | |
98 | ||
99 | /** ロガー。 */ | |
100 | 1 | private static final Logger logger = Logger.getLogger(Forward.class); |
101 | ||
102 | /** 空のパラメータ。 */ | |
103 | 1 | private static final Map<String, String[]> EMPTY_PARAMETERS = Collections |
104 | .emptyMap(); | |
105 | ||
106 | /** フォワード先のパス。 */ | |
107 | private String path; | |
108 | ||
109 | /** ルーティング。 */ | |
110 | private final Map<String, Routing> routings; | |
111 | ||
112 | /** フォワード先のアクションクラス */ | |
113 | private Class<? extends Action> actionClass; | |
114 | ||
115 | /** フォワード先のアクションクラスのメソッド名 */ | |
116 | private String methodName; | |
117 | ||
118 | /** フォワード時のパラメータ */ | |
119 | private Map<String, String[]> parameters; | |
120 | ||
121 | /** | |
122 | * インスタンスを生成します。 | |
123 | * | |
124 | * @param path | |
125 | * フォワード先のパス | |
126 | */ | |
127 | 8 | public Forward(final String path) { |
128 | 8 | this.path = path; |
129 | 8 | this.routings = null; |
130 | 8 | } |
131 | ||
132 | /** | |
133 | * インスタンスを生成します。 | |
134 | * | |
135 | * @param actionClass | |
136 | * アクションクラス | |
137 | * @param methodName | |
138 | * アクションメソッド名 | |
139 | * @param parameters | |
140 | * パラメータ | |
141 | * @since 1.1.0 | |
142 | */ | |
143 | public Forward(final Class<? extends Action> actionClass, | |
144 | 3 | final String methodName, final Map<String, String[]> parameters) { |
145 | 3 | this.actionClass = actionClass; |
146 | 3 | this.methodName = methodName; |
147 | 3 | this.parameters = parameters; |
148 | 3 | final Method method = ClassUtil.getMethod(actionClass, methodName, |
149 | new Class[0]); | |
150 | 3 | final Routing routing = new ForwardRouting(actionClass, method); |
151 | 3 | this.routings = Collections.singletonMap(null, routing); |
152 | 3 | } |
153 | ||
154 | /** | |
155 | * 指定されたアクションクラスのindexメソッドへフォワードするインスタンスを生成します。 | |
156 | * | |
157 | * @param actionClass | |
158 | * アクションクラス | |
159 | * @since 1.1.0 | |
160 | */ | |
161 | public Forward(final Class<? extends Action> actionClass) { | |
162 | 1 | this(actionClass, "index"); |
163 | 1 | } |
164 | ||
165 | /** | |
166 | * 指定されたアクションメソッドへフォワードするインスタンスを生成します。 | |
167 | * | |
168 | * @param actionClass | |
169 | * アクションクラス | |
170 | * @param methodName | |
171 | * アクションメソッド名 | |
172 | * @since 1.1.0 | |
173 | */ | |
174 | public Forward(final Class<? extends Action> actionClass, | |
175 | final String methodName) { | |
176 | 3 | this(actionClass, methodName, EMPTY_PARAMETERS); |
177 | 3 | } |
178 | ||
179 | /** | |
180 | * パスを取得します。 | |
181 | * | |
182 | * @param characterEncoding | |
183 | * URI のエンコーディング | |
184 | * @return パス | |
185 | */ | |
186 | public String getPath(final String characterEncoding) { | |
187 | 16 | if (isReverseLookupRedirect()) { |
188 | 6 | final S2Container container = SingletonS2ContainerFactory |
189 | .getContainer(); | |
190 | 6 | final PathResolver pathResolver = PathResolver.class.cast(container |
191 | .getComponent(PathResolver.class)); | |
192 | 6 | this.path = pathResolver.buildInternalForwardPath(parameters, |
193 | characterEncoding); | |
194 | } | |
195 | 16 | return this.path; |
196 | } | |
197 | ||
198 | /** | |
199 | * アクションクラスを指定したフォワードかどうかを判定します。 | |
200 | * | |
201 | * @return アクションクラスを指定したフォワードならtrue | |
202 | */ | |
203 | private boolean isReverseLookupRedirect() { | |
204 | 20 | return this.actionClass != null && this.methodName != null |
205 | && this.parameters != null; | |
206 | } | |
207 | ||
208 | /** | |
209 | * {@inheritDoc} | |
210 | */ | |
211 | public void execute(final Action action, | |
212 | final Class<? extends Action> actionClass, final Method method, | |
213 | final HttpServletRequest request, final HttpServletResponse response) | |
214 | throws ServletException, IOException { | |
215 | 6 | action.invokePreRenderMethod(method); |
216 | ||
217 | 6 | final String forwardPath = calculateForwardPath(getPath(request |
218 | .getCharacterEncoding()), actionClass, request | |
219 | .getCharacterEncoding()); | |
220 | 6 | if (this.routings != null) { |
221 | 3 | request.setAttribute(ATTR_ROUTINGS, this.routings); |
222 | } | |
223 | 6 | if (logger.isDebugEnabled()) { |
224 | 6 | logger.log("DCUB0001", new Object[] { forwardPath, routings }); |
225 | } | |
226 | 6 | final RequestDispatcher dispatcher = request |
227 | .getRequestDispatcher(forwardPath); | |
228 | 6 | dispatcher.forward(request, response); |
229 | 6 | if (logger.isDebugEnabled()) { |
230 | 6 | logger.log("DCUB0002", new Object[] { forwardPath }); |
231 | } | |
232 | ||
233 | 6 | action.invokePostRenderMethod(method); |
234 | 6 | if (action.getFlash() != null) { |
235 | 6 | action.getFlash().clear(); |
236 | } | |
237 | 6 | } |
238 | ||
239 | /** | |
240 | * フォワードするパスを計算します。 | |
241 | * | |
242 | * @param actionClass | |
243 | * アクションクラス | |
244 | * @param characterEncoding | |
245 | * URI のエンコーディング | |
246 | * @return フォワードするパス | |
247 | */ | |
248 | protected String calculateForwardPath(final String path, | |
249 | final Class<? extends Action> actionClass, | |
250 | final String characterEncoding) { | |
251 | final String absolutePath; | |
252 | 6 | if (getPath(characterEncoding).startsWith("/")) { |
253 | 4 | absolutePath = path; |
254 | } else { | |
255 | 2 | final String actionDirectory = CubbyUtils |
256 | .getActionDirectory(actionClass); | |
257 | 2 | if (StringUtil.isEmpty(actionDirectory)) { |
258 | 0 | absolutePath = "/" + path; |
259 | } else { | |
260 | 2 | final StringBuilder builder = new StringBuilder(); |
261 | 2 | if (!actionDirectory.startsWith("/")) { |
262 | 2 | builder.append("/"); |
263 | } | |
264 | 2 | builder.append(actionDirectory); |
265 | 2 | if (!actionDirectory.endsWith("/")) { |
266 | 2 | builder.append("/"); |
267 | } | |
268 | 2 | builder.append(path); |
269 | 2 | absolutePath = builder.toString(); |
270 | } | |
271 | } | |
272 | 6 | return absolutePath; |
273 | } | |
274 | ||
275 | /** | |
276 | * パラメータを追加します。 | |
277 | * | |
278 | * @param paramName | |
279 | * パラメータ名 | |
280 | * @param paramValue | |
281 | * パラメータの値。{@code Object#toString()}の結果が値として使用されます。 | |
282 | * @return フォワードする URL | |
283 | */ | |
284 | public Forward param(String paramName, Object paramValue) { | |
285 | 4 | return param(paramName, new String[] { paramValue.toString() }); |
286 | } | |
287 | ||
288 | /** | |
289 | * パラメータを追加します。 | |
290 | * | |
291 | * @param paramName | |
292 | * パラメータ名 | |
293 | * @param paramValues | |
294 | * パラメータの値の配列。配列の要素の{@code Object#toString()}の結果がそれぞれの値として使用されます。 | |
295 | * @return フォワードする URL | |
296 | */ | |
297 | public Forward param(final String paramName, final Object[] paramValues) { | |
298 | 0 | return param(paramName, toStringArray(paramValues)); |
299 | } | |
300 | ||
301 | /** | |
302 | * パラメータを追加します。 | |
303 | * | |
304 | * @param paramName | |
305 | * パラメータ名 | |
306 | * @param paramValues | |
307 | * パラメータの値 | |
308 | * @return フォワードする URL | |
309 | */ | |
310 | public Forward param(final String paramName, final String[] paramValues) { | |
311 | 4 | if (isReverseLookupRedirect()) { |
312 | 2 | if (this.parameters == EMPTY_PARAMETERS) { |
313 | 1 | this.parameters = new HashMap<String, String[]>(); |
314 | } | |
315 | 2 | this.parameters.put(paramName, paramValues); |
316 | } else { | |
317 | 2 | QueryStringBuilder builder = new QueryStringBuilder(this.path); |
318 | 2 | builder.addParam(paramName, paramValues); |
319 | 2 | this.path = builder.toString(); |
320 | } | |
321 | 4 | return this; |
322 | } | |
323 | ||
324 | /** | |
325 | * {@code Object#toString()}型の配列を{@code Object#toString()}型の配列に変換します。 | |
326 | * <p> | |
327 | * 配列のそれぞれの要素に対して{@code Object#toString()}を使用して変換します。 | |
328 | * </p> | |
329 | * | |
330 | * @param paramValues | |
331 | * {@code Object#toString()}型の配列 | |
332 | * @return {@code Object#toString()}型の配列。 | |
333 | */ | |
334 | private String[] toStringArray(final Object[] paramValues) { | |
335 | 0 | String[] values = new String[paramValues.length]; |
336 | 0 | for (int i = 0; i < paramValues.length; i++) { |
337 | 0 | values[i] = paramValues[i].toString(); |
338 | } | |
339 | 0 | return values; |
340 | } | |
341 | ||
342 | /** | |
343 | * パスを取得します。 | |
344 | * | |
345 | * @return パス | |
346 | * @deprecated use {@link #getPath(String)} | |
347 | */ | |
348 | @Deprecated | |
349 | public String getPath() { | |
350 | 0 | return getPath("UTF-8"); |
351 | } | |
352 | ||
353 | /** | |
354 | * アクションメソッドへフォワードするためのルーティング。 | |
355 | * | |
356 | * @author baba | |
357 | * @since 1.1.0 | |
358 | */ | |
359 | 3 | private static class ForwardRouting implements Routing { |
360 | ||
361 | /** アクションクラス。 */ | |
362 | private Class<? extends Action> actionClass; | |
363 | ||
364 | /** アクションメソッド。 */ | |
365 | private Method method; | |
366 | ||
367 | /** | |
368 | * {@inheritDoc} | |
369 | */ | |
370 | private ForwardRouting(final Class<? extends Action> actionClass, | |
371 | 3 | final Method method) { |
372 | 3 | this.actionClass = actionClass; |
373 | 3 | this.method = method; |
374 | 3 | } |
375 | ||
376 | /** | |
377 | * {@inheritDoc} | |
378 | */ | |
379 | public Class<? extends Action> getActionClass() { | |
380 | 3 | return actionClass; |
381 | } | |
382 | ||
383 | /** | |
384 | * {@inheritDoc} | |
385 | */ | |
386 | public Method getMethod() { | |
387 | 3 | return method; |
388 | } | |
389 | ||
390 | /** | |
391 | * {@inheritDoc} | |
392 | */ | |
393 | public String getActionPath() { | |
394 | 0 | return null; |
395 | } | |
396 | ||
397 | /** | |
398 | * {@inheritDoc} | |
399 | */ | |
400 | public List<String> getUriParameterNames() { | |
401 | 0 | return null; |
402 | } | |
403 | ||
404 | /** | |
405 | * {@inheritDoc} | |
406 | */ | |
407 | public Pattern getPattern() { | |
408 | 0 | return null; |
409 | } | |
410 | ||
411 | /** | |
412 | * {@inheritDoc} | |
413 | */ | |
414 | public RequestMethod getRequestMethod() { | |
415 | 0 | return null; |
416 | } | |
417 | ||
418 | /** | |
419 | * {@inheritDoc} | |
420 | */ | |
421 | public String getOnSubmit() { | |
422 | 0 | return null; |
423 | } | |
424 | ||
425 | /** | |
426 | * {@inheritDoc} | |
427 | */ | |
428 | public int getPriority() { | |
429 | 0 | return 0; |
430 | } | |
431 | ||
432 | /** | |
433 | * {@inheritDoc} | |
434 | */ | |
435 | public boolean isAuto() { | |
436 | 0 | return false; |
437 | } | |
438 | ||
439 | /** | |
440 | * {@inheritDoc} | |
441 | */ | |
442 | public boolean isAcceptable(final String requestMethod) { | |
443 | 0 | return true; |
444 | } | |
445 | ||
446 | /** | |
447 | * {@inheritDoc} | |
448 | */ | |
449 | @Override | |
450 | public String toString() { | |
451 | 3 | return new StringBuilder().append("[").append(method).append("]") |
452 | .toString(); | |
453 | } | |
454 | ||
455 | } | |
456 | ||
457 | } |