1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19 package org.apache.shiro.web.env;
20
21 import org.apache.shiro.config.ConfigurationException;
22 import org.apache.shiro.config.Ini;
23 import org.apache.shiro.ini.IniFactorySupport;
24 import org.apache.shiro.lang.io.ResourceUtils;
25 import org.apache.shiro.lang.util.Destroyable;
26 import org.apache.shiro.lang.util.Factory;
27 import org.apache.shiro.lang.util.Initializable;
28 import org.apache.shiro.lang.util.StringUtils;
29 import org.apache.shiro.util.CollectionUtils;
30 import org.apache.shiro.web.config.IniFilterChainResolverFactory;
31 import org.apache.shiro.web.config.ShiroFilterConfiguration;
32 import org.apache.shiro.web.config.WebIniSecurityManagerFactory;
33 import org.apache.shiro.web.filter.mgt.FilterChainResolver;
34 import org.apache.shiro.web.mgt.WebSecurityManager;
35 import org.apache.shiro.web.util.WebUtils;
36 import org.slf4j.Logger;
37 import org.slf4j.LoggerFactory;
38
39 import javax.servlet.ServletContext;
40 import java.io.IOException;
41 import java.io.InputStream;
42 import java.util.HashMap;
43 import java.util.Map;
44
45
46
47
48
49
50 public class IniWebEnvironment extends ResourceBasedWebEnvironment implements Initializable, Destroyable {
51
52
53
54
55 public static final String DEFAULT_WEB_INI_RESOURCE_PATH = "/WEB-INF/shiro.ini";
56
57
58
59 public static final String FILTER_CHAIN_RESOLVER_NAME = "filterChainResolver";
60
61
62
63
64 public static final String SHIRO_FILTER_CONFIG_NAME = "shiroFilter";
65
66 private static final Logger LOGGER = LoggerFactory.getLogger(IniWebEnvironment.class);
67
68
69
70
71 private Ini ini;
72
73 @SuppressWarnings("deprecation")
74 private WebIniSecurityManagerFactory factory;
75
76 @SuppressWarnings("deprecation")
77 public IniWebEnvironment() {
78 factory = new WebIniSecurityManagerFactory();
79 }
80
81
82
83
84
85 public void init() {
86
87 setIni(parseConfig());
88
89 configure();
90 }
91
92
93
94
95
96
97
98
99
100 protected Ini parseConfig() {
101 Ini ini = getIni();
102
103 String[] configLocations = getConfigLocations();
104
105 if (LOGGER.isWarnEnabled() && !CollectionUtils.isEmpty(ini)
106 && configLocations != null && configLocations.length > 0) {
107 LOGGER.warn("Explicit INI instance has been provided, but configuration locations have also been "
108 + "specified. The {} implementation does not currently support multiple Ini config, but this may "
109 + "be supported in the future. Only the INI instance will be used for configuration.",
110 IniWebEnvironment.class.getName());
111 }
112
113 if (CollectionUtils.isEmpty(ini)) {
114 LOGGER.debug("Checking any specified config locations.");
115 ini = getSpecifiedIni(configLocations);
116 }
117
118 if (CollectionUtils.isEmpty(ini)) {
119 LOGGER.debug("No INI instance or config locations specified. Trying default config locations.");
120 ini = getDefaultIni();
121 }
122
123
124
125 ini = mergeIni(getFrameworkIni(), ini);
126
127 if (CollectionUtils.isEmpty(ini)) {
128 String msg = "Shiro INI configuration was either not found or discovered to be empty/unconfigured.";
129 throw new ConfigurationException(msg);
130 }
131 return ini;
132 }
133
134 protected void configure() {
135
136 this.objects.clear();
137
138 WebSecurityManager securityManager = createWebSecurityManager();
139 setWebSecurityManager(securityManager);
140
141 ShiroFilterConfiguration filterConfiguration = createFilterConfiguration();
142 setShiroFilterConfiguration(filterConfiguration);
143
144 FilterChainResolver resolver = createFilterChainResolver();
145 if (resolver != null) {
146 setFilterChainResolver(resolver);
147 }
148 }
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191 protected Ini getFrameworkIni() {
192 return null;
193 }
194
195 protected Ini getSpecifiedIni(String[] configLocations) throws ConfigurationException {
196
197 Ini ini = null;
198
199 if (configLocations != null && configLocations.length > 0) {
200
201 if (configLocations.length > 1) {
202 LOGGER.warn("More than one Shiro .ini config location has been specified. Only the first will be "
203 + "used for configuration as the {} implementation does not currently support multiple "
204 + "files. This may be supported in the future however.", IniWebEnvironment.class.getName());
205 }
206
207
208 ini = createIni(configLocations[0], true);
209 }
210
211 return ini;
212 }
213
214 protected Ini mergeIni(Ini ini1, Ini ini2) {
215
216 if (ini1 == null) {
217 return ini2;
218 }
219
220 if (ini2 == null) {
221 return ini1;
222 }
223
224
225 Ini iniResult = new Ini(ini1);
226 iniResult.merge(ini2);
227
228 return iniResult;
229 }
230
231 protected Ini getDefaultIni() {
232
233 Ini ini = null;
234
235 String[] configLocations = getDefaultConfigLocations();
236 if (configLocations != null) {
237 for (String location : configLocations) {
238 ini = createIni(location, false);
239 if (!CollectionUtils.isEmpty(ini)) {
240 LOGGER.debug("Discovered non-empty INI configuration at location '{}'. Using for configuration.",
241 location);
242 break;
243 }
244 }
245 }
246
247 return ini;
248 }
249
250
251
252
253
254
255
256
257
258
259
260
261
262 protected Ini createIni(String configLocation, boolean required) throws ConfigurationException {
263
264 Ini ini = null;
265
266 if (configLocation != null) {
267 ini = convertPathToIni(configLocation, required);
268 }
269 if (required && CollectionUtils.isEmpty(ini)) {
270 String msg = "Required configuration location '" + configLocation + "' does not exist or did not "
271 + "contain any INI configuration.";
272 throw new ConfigurationException(msg);
273 }
274
275 return ini;
276 }
277
278
279 protected ShiroFilterConfiguration createFilterConfiguration() {
280 return (ShiroFilterConfiguration) this.objects.get(SHIRO_FILTER_CONFIG_NAME);
281 }
282
283 @SuppressWarnings("deprecation")
284 protected FilterChainResolver createFilterChainResolver() {
285
286 FilterChainResolver resolver = null;
287
288 Ini ini = getIni();
289
290 if (!CollectionUtils.isEmpty(ini)) {
291 @SuppressWarnings("unchecked")
292 Factory<FilterChainResolver> factory = (Factory<FilterChainResolver>) this.objects.get(FILTER_CHAIN_RESOLVER_NAME);
293 if (factory instanceof IniFactorySupport) {
294 var iniFactory = (IniFactorySupport<?>) factory;
295 iniFactory.setIni(ini);
296 iniFactory.setDefaults(this.objects);
297 }
298 resolver = factory.getInstance();
299 }
300
301 return resolver;
302 }
303
304 protected WebSecurityManager createWebSecurityManager() {
305
306 Ini ini = getIni();
307 if (!CollectionUtils.isEmpty(ini)) {
308 factory.setIni(ini);
309 }
310
311 Map<String, Object> defaults = getDefaults();
312 if (!CollectionUtils.isEmpty(defaults)) {
313 factory.setDefaults(defaults);
314 }
315
316 WebSecurityManager wsm = (WebSecurityManager) factory.getInstance();
317
318
319
320 Map<String, ?> beans = factory.getBeans();
321 if (!CollectionUtils.isEmpty(beans)) {
322 this.objects.putAll(beans);
323 }
324
325 return wsm;
326 }
327
328
329
330
331
332
333 @SuppressWarnings("deprecation")
334 protected String[] getDefaultConfigLocations() {
335 return new String[] {
336 DEFAULT_WEB_INI_RESOURCE_PATH,
337 IniFactorySupport.DEFAULT_INI_RESOURCE_PATH
338 };
339 }
340
341
342
343
344
345
346
347
348
349
350
351
352 private Ini convertPathToIni(String path, boolean required) {
353
354
355
356 Ini ini = null;
357
358 if (StringUtils.hasText(path)) {
359 InputStream is = null;
360
361
362 if (!ResourceUtils.hasResourcePrefix(path)) {
363 is = getServletContextResourceStream(path);
364 } else {
365 try {
366 is = ResourceUtils.getInputStreamForPath(path);
367 } catch (IOException e) {
368 if (required) {
369 throw new ConfigurationException(e);
370 } else {
371 if (LOGGER.isDebugEnabled()) {
372 LOGGER.debug("Unable to load optional path '" + path + "'.", e);
373 }
374 }
375 }
376 }
377 if (is != null) {
378 ini = new Ini();
379 ini.load(is);
380 } else {
381 if (required) {
382 throw new ConfigurationException("Unable to load resource path '" + path + "'");
383 }
384 }
385 }
386
387 return ini;
388 }
389
390
391 private InputStream getServletContextResourceStream(String path) {
392 InputStream is = null;
393
394 path = WebUtils.normalize(path);
395 ServletContext sc = getServletContext();
396 if (sc != null) {
397 is = sc.getResourceAsStream(path);
398 }
399
400 return is;
401 }
402
403
404
405
406
407
408 public Ini getIni() {
409 return this.ini;
410 }
411
412
413
414
415
416
417
418
419
420 public void setIni(Ini ini) {
421 this.ini = ini;
422 }
423
424 protected Map<String, Object> getDefaults() {
425 Map<String, Object> defaults = new HashMap<String, Object>();
426 defaults.put(FILTER_CHAIN_RESOLVER_NAME, new IniFilterChainResolverFactory());
427 defaults.put(SHIRO_FILTER_CONFIG_NAME, new ShiroFilterConfiguration());
428 return defaults;
429 }
430
431
432
433
434
435
436
437 @SuppressWarnings({"unused", "deprecation"})
438 protected WebIniSecurityManagerFactory getSecurityManagerFactory() {
439 return factory;
440 }
441
442
443
444
445
446
447
448 @SuppressWarnings("deprecation")
449 protected void setSecurityManagerFactory(WebIniSecurityManagerFactory factory) {
450 this.factory = factory;
451 }
452 }