001/* 002 Licensed to the Apache Software Foundation (ASF) under one 003 or more contributor license agreements. See the NOTICE file 004 distributed with this work for additional information 005 regarding copyright ownership. The ASF licenses this file 006 to you under the Apache License, Version 2.0 (the 007 "License"); you may not use this file except in compliance 008 with the License. You may obtain a copy of the License at 009 010 http://www.apache.org/licenses/LICENSE-2.0 011 012 Unless required by applicable law or agreed to in writing, 013 software distributed under the License is distributed on an 014 "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 015 KIND, either express or implied. See the License for the 016 specific language governing permissions and limitations 017 under the License. 018 */ 019package org.apache.wiki; 020 021import org.apache.commons.lang3.StringUtils; 022import org.apache.commons.lang3.time.StopWatch; 023import org.apache.log4j.Logger; 024import org.apache.log4j.PropertyConfigurator; 025import org.apache.wiki.api.engine.AdminBeanManager; 026import org.apache.wiki.api.engine.FilterManager; 027import org.apache.wiki.api.engine.PluginManager; 028import org.apache.wiki.api.exceptions.FilterException; 029import org.apache.wiki.api.exceptions.NoRequiredPropertyException; 030import org.apache.wiki.api.exceptions.NoSuchVariableException; 031import org.apache.wiki.api.exceptions.ProviderException; 032import org.apache.wiki.api.exceptions.WikiException; 033import org.apache.wiki.attachment.Attachment; 034import org.apache.wiki.attachment.AttachmentManager; 035import org.apache.wiki.auth.AuthenticationManager; 036import org.apache.wiki.auth.AuthorizationManager; 037import org.apache.wiki.auth.UserManager; 038import org.apache.wiki.auth.acl.AclManager; 039import org.apache.wiki.auth.authorize.GroupManager; 040import org.apache.wiki.content.PageRenamer; 041import org.apache.wiki.diff.DifferenceManager; 042import org.apache.wiki.event.WikiEngineEvent; 043import org.apache.wiki.event.WikiEventListener; 044import org.apache.wiki.event.WikiEventManager; 045import org.apache.wiki.event.WikiPageEvent; 046import org.apache.wiki.event.WikiPageRenameEvent; 047import org.apache.wiki.i18n.InternationalizationManager; 048import org.apache.wiki.pages.PageManager; 049import org.apache.wiki.pages.PageTimeComparator; 050import org.apache.wiki.parser.MarkupParser; 051import org.apache.wiki.parser.WikiDocument; 052import org.apache.wiki.providers.WikiPageProvider; 053import org.apache.wiki.render.RenderingManager; 054import org.apache.wiki.rss.RSSGenerator; 055import org.apache.wiki.rss.RSSThread; 056import org.apache.wiki.search.SearchManager; 057import org.apache.wiki.tasks.TasksManager; 058import org.apache.wiki.ui.Command; 059import org.apache.wiki.ui.CommandResolver; 060import org.apache.wiki.ui.EditorManager; 061import org.apache.wiki.ui.TemplateManager; 062import org.apache.wiki.ui.progress.ProgressManager; 063import org.apache.wiki.url.URLConstructor; 064import org.apache.wiki.util.ClassUtil; 065import org.apache.wiki.util.PropertyReader; 066import org.apache.wiki.util.TextUtil; 067import org.apache.wiki.workflow.Decision; 068import org.apache.wiki.workflow.DecisionRequiredException; 069import org.apache.wiki.workflow.Fact; 070import org.apache.wiki.workflow.Step; 071import org.apache.wiki.workflow.Workflow; 072import org.apache.wiki.workflow.WorkflowBuilder; 073import org.apache.wiki.workflow.WorkflowManager; 074 075import javax.servlet.ServletConfig; 076import javax.servlet.ServletContext; 077import javax.servlet.http.HttpServletRequest; 078import java.io.File; 079import java.io.IOException; 080import java.io.UnsupportedEncodingException; 081import java.net.MalformedURLException; 082import java.net.URL; 083import java.net.URLDecoder; 084import java.net.URLEncoder; 085import java.nio.charset.Charset; 086import java.nio.charset.StandardCharsets; 087import java.security.Principal; 088import java.util.ArrayList; 089import java.util.Collection; 090import java.util.Date; 091import java.util.Enumeration; 092import java.util.List; 093import java.util.Locale; 094import java.util.Map; 095import java.util.Properties; 096import java.util.Set; 097import java.util.TimeZone; 098import java.util.TreeSet; 099import java.util.concurrent.ConcurrentHashMap; 100 101 102/** 103 * Provides Wiki services to the JSP page. 104 * 105 * <P> 106 * This is the main interface through which everything should go. 107 * 108 * <P> 109 * Using this class: Always get yourself an instance from JSP page 110 * by using the WikiEngine.getInstance() method. Never create a new 111 * WikiEngine() from scratch, unless you're writing tests. 112 * <p> 113 * There's basically only a single WikiEngine for each web application, and 114 * you should always get it using the WikiEngine.getInstance() method. 115 */ 116public class WikiEngine 117{ 118 private static final String ATTR_WIKIENGINE = "org.apache.wiki.WikiEngine"; 119 120 private static final Logger log = Logger.getLogger(WikiEngine.class); 121 122 /** True, if log4j has been configured. */ 123 // FIXME: If you run multiple applications, the first application 124 // to run defines where the log goes. Not what we want. 125 private static boolean c_configured = false; 126 127 /** Stores properties. */ 128 private Properties m_properties; 129 130 /** The default inlining pattern. Currently "*.png" */ 131 public static final String DEFAULT_INLINEPATTERN = "*.png"; 132 133 /** The name used for the default template. The value is {@value}. */ 134 public static final String DEFAULT_TEMPLATE_NAME = "default"; 135 136 /** Property for application name */ 137 public static final String PROP_APPNAME = "jspwiki.applicationName"; 138 139 /** This property defines the inline image pattern. It's current value is {@value} */ 140 public static final String PROP_INLINEIMAGEPTRN = "jspwiki.translatorReader.inlinePattern"; 141 142 /** Property start for any interwiki reference. */ 143 public static final String PROP_INTERWIKIREF = "jspwiki.interWikiRef."; 144 145 /** If true, then the user name will be stored with the page data.*/ 146 public static final String PROP_STOREUSERNAME= "jspwiki.storeUserName"; 147 148 /** Define the used encoding. Currently supported are ISO-8859-1 and UTF-8 */ 149 public static final String PROP_ENCODING = "jspwiki.encoding"; 150 151 /** Do not use encoding in WikiJSPFilter, default is false for most servers. 152 Double negative, cause for most servers you don't need the property */ 153 public static final String PROP_NO_FILTER_ENCODING = "jspwiki.nofilterencoding"; 154 155 /** The name for the property which allows you to set the current reference 156 * style. The value is {@value}. 157 */ 158 public static final String PROP_REFSTYLE = "jspwiki.referenceStyle"; 159 160 /** Property name for the "spaces in titles" -hack. */ 161 public static final String PROP_BEAUTIFYTITLE = "jspwiki.breakTitleWithSpaces"; 162 163 /** Property name for where the jspwiki work directory should be. 164 If not specified, reverts to ${java.tmpdir}. */ 165 public static final String PROP_WORKDIR = "jspwiki.workDir"; 166 167 /** The name of the cookie that gets stored to the user browser. */ 168 public static final String PREFS_COOKIE_NAME = "JSPWikiUserProfile"; 169 170 /** Property name for the "match english plurals" -hack. */ 171 public static final String PROP_MATCHPLURALS = "jspwiki.translatorReader.matchEnglishPlurals"; 172 173 /** Property name for the template that is used. */ 174 public static final String PROP_TEMPLATEDIR = "jspwiki.templateDir"; 175 176 /** Property name for the default front page. */ 177 public static final String PROP_FRONTPAGE = "jspwiki.frontPage"; 178 179 /** Property name for setting the url generator instance */ 180 181 public static final String PROP_URLCONSTRUCTOR = "jspwiki.urlConstructor"; 182 183 /** If this property is set to false, all filters are disabled when translating. */ 184 public static final String PROP_RUNFILTERS = "jspwiki.runFilters"; 185 186 /** Does the work in renaming pages. */ 187 private PageRenamer m_pageRenamer = null; 188 189 /** The name of the property containing the ACLManager implementing class. 190 * The value is {@value}. */ 191 public static final String PROP_ACL_MANAGER_IMPL = "jspwiki.aclManager"; 192 193 /** If this property is set to false, we don't allow the creation of empty pages */ 194 public static final String PROP_ALLOW_CREATION_OF_EMPTY_PAGES = "jspwiki.allowCreationOfEmptyPages"; 195 196 /** Should the user info be saved with the page data as well? */ 197 private boolean m_saveUserInfo = true; 198 199 /** If true, uses UTF8 encoding for all data */ 200 private boolean m_useUTF8 = true; 201 202 /** Store the file path to the basic URL. When we're not running as 203 a servlet, it defaults to the user's current directory. */ 204 private String m_rootPath = System.getProperty("user.dir"); 205 206 /** Stores references between wikipages. */ 207 private ReferenceManager m_referenceManager = null; 208 209 /** Stores the Plugin manager */ 210 private PluginManager m_pluginManager; 211 212 /** Stores the Variable manager */ 213 private VariableManager m_variableManager; 214 215 /** Stores the Attachment manager */ 216 private AttachmentManager m_attachmentManager = null; 217 218 /** Stores the Page manager */ 219 private PageManager m_pageManager = null; 220 221 /** Stores the authorization manager */ 222 private AuthorizationManager m_authorizationManager = null; 223 224 /** Stores the authentication manager.*/ 225 private AuthenticationManager m_authenticationManager = null; 226 227 /** Stores the ACL manager. */ 228 private AclManager m_aclManager = null; 229 230 /** Resolves wiki actions, JSPs and special pages. */ 231 private CommandResolver m_commandResolver = null; 232 233 private TemplateManager m_templateManager = null; 234 235 /** Does all our diffs for us. */ 236 private DifferenceManager m_differenceManager; 237 238 /** Handlers page filters. */ 239 private FilterManager m_filterManager; 240 241 /** Stores the Search manager */ 242 private SearchManager m_searchManager = null; 243 244 /** Facade for managing users */ 245 private UserManager m_userManager = null; 246 247 /** Facade for managing users */ 248 private GroupManager m_groupManager = null; 249 250 private RenderingManager m_renderingManager; 251 252 private EditorManager m_editorManager; 253 254 private InternationalizationManager m_internationalizationManager; 255 256 private ProgressManager m_progressManager; 257 258 private TasksManager m_tasksManager; 259 260 /** Constructs URLs */ 261 private URLConstructor m_urlConstructor; 262 263 /** Generates RSS feed when requested. */ 264 private RSSGenerator m_rssGenerator; 265 266 /** The RSS file to generate. */ 267 private String m_rssFile; 268 269 /** Store the ServletContext that we're in. This may be null if WikiEngine 270 is not running inside a servlet container (i.e. when testing). */ 271 private ServletContext m_servletContext = null; 272 273 /** If true, all titles will be cleaned. */ 274 private boolean m_beautifyTitle = false; 275 276 /** Stores the template path. This is relative to "templates". */ 277 private String m_templateDir; 278 279 /** The default front page name. Defaults to "Main". */ 280 private String m_frontPage; 281 282 /** The time when this engine was started. */ 283 private Date m_startTime; 284 285 /** The location where the work directory is. */ 286 private String m_workDir; 287 288 /** Each engine has their own application id. */ 289 private String m_appid = ""; 290 291 private boolean m_isConfigured = false; // Flag. 292 293 294 /** Each engine has its own workflow manager. */ 295 private WorkflowManager m_workflowMgr = null; 296 297 private AdminBeanManager m_adminBeanManager; 298 299 /** Stores wikiengine attributes. */ 300 private Map<String,Object> m_attributes = new ConcurrentHashMap<>(); 301 302 /** 303 * Gets a WikiEngine related to this servlet. Since this method 304 * is only called from JSP pages (and JspInit()) to be specific, 305 * we throw a RuntimeException if things don't work. 306 * 307 * @param config The ServletConfig object for this servlet. 308 * 309 * @return A WikiEngine instance. 310 * @throws InternalWikiException in case something fails. This 311 * is a RuntimeException, so be prepared for it. 312 */ 313 314 // FIXME: It seems that this does not work too well, jspInit() 315 // does not react to RuntimeExceptions, or something... 316 317 public static synchronized WikiEngine getInstance( ServletConfig config ) 318 throws InternalWikiException 319 { 320 return getInstance( config.getServletContext(), null ); 321 } 322 323 /** 324 * Gets a WikiEngine related to the servlet. Works like getInstance(ServletConfig), 325 * but does not force the Properties object. This method is just an optional way 326 * of initializing a WikiEngine for embedded JSPWiki applications; normally, you 327 * should use getInstance(ServletConfig). 328 * 329 * @param config The ServletConfig of the webapp servlet/JSP calling this method. 330 * @param props A set of properties, or null, if we are to load JSPWiki's default 331 * jspwiki.properties (this is the usual case). 332 * 333 * @return One well-behaving WikiEngine instance. 334 */ 335 public static synchronized WikiEngine getInstance( ServletConfig config, 336 Properties props ) 337 { 338 return getInstance( config.getServletContext(), props ); 339 } 340 341 /** 342 * Gets a WikiEngine related to the servlet. Works just like getInstance( ServletConfig ) 343 * 344 * @param context The ServletContext of the webapp servlet/JSP calling this method. 345 * @param props A set of properties, or null, if we are to load JSPWiki's default 346 * jspwiki.properties (this is the usual case). 347 * 348 * @return One fully functional, properly behaving WikiEngine. 349 * @throws InternalWikiException If the WikiEngine instantiation fails. 350 */ 351 352 // FIXME: Potential make-things-easier thingy here: no need to fetch the wikiengine anymore 353 // Wiki.jsp.jspInit() [really old code]; it's probably even faster to fetch it 354 // using this method every time than go to pageContext.getAttribute(). 355 356 public static synchronized WikiEngine getInstance( ServletContext context, 357 Properties props ) 358 throws InternalWikiException 359 { 360 WikiEngine engine = (WikiEngine) context.getAttribute( ATTR_WIKIENGINE ); 361 362 if( engine == null ) 363 { 364 String appid = Integer.toString(context.hashCode()); //FIXME: Kludge, use real type. 365 366 context.log(" Assigning new engine to "+appid); 367 try 368 { 369 if( props == null ) 370 { 371 props = PropertyReader.loadWebAppProps( context ); 372 } 373 374 engine = new WikiEngine( context, appid, props ); 375 context.setAttribute( ATTR_WIKIENGINE, engine ); 376 } 377 catch( Exception e ) 378 { 379 context.log( "ERROR: Failed to create a Wiki engine: "+e.getMessage() ); 380 log.error( "ERROR: Failed to create a Wiki engine, stacktrace follows " , e); 381 throw new InternalWikiException( "No wiki engine, check logs." , e); 382 } 383 384 } 385 386 return engine; 387 } 388 389 390 /** 391 * Instantiate the WikiEngine using a given set of properties. 392 * Use this constructor for testing purposes only. 393 * 394 * @param properties A set of properties to use to initialize this WikiEngine. 395 * @throws WikiException If the initialization fails. 396 */ 397 public WikiEngine( Properties properties ) 398 throws WikiException 399 { 400 initialize( properties ); 401 } 402 403 /** 404 * Instantiate using this method when you're running as a servlet and 405 * WikiEngine will figure out where to look for the property 406 * file. 407 * Do not use this method - use WikiEngine.getInstance() instead. 408 * 409 * @param context A ServletContext. 410 * @param appid An Application ID. This application is an unique random string which 411 * is used to recognize this WikiEngine. 412 * @param props The WikiEngine configuration. 413 * @throws WikiException If the WikiEngine construction fails. 414 */ 415 protected WikiEngine( ServletContext context, String appid, Properties props ) 416 throws WikiException 417 { 418 super(); 419 m_servletContext = context; 420 m_appid = appid; 421 422 // Stash the WikiEngine in the servlet context 423 if ( context != null ) 424 { 425 context.setAttribute( ATTR_WIKIENGINE, this ); 426 m_rootPath = context.getRealPath("/"); 427 } 428 429 try 430 { 431 // 432 // Note: May be null, if JSPWiki has been deployed in a WAR file. 433 // 434 initialize( props ); 435 log.info("Root path for this Wiki is: '"+m_rootPath+"'"); 436 } 437 catch( Exception e ) 438 { 439 String msg = Release.APPNAME+": Unable to load and setup properties from jspwiki.properties. "+e.getMessage(); 440 if ( context != null ) 441 { 442 context.log( msg ); 443 } 444 throw new WikiException( msg, e ); 445 } 446 } 447 448 /** 449 * Does all the real initialization. 450 */ 451 private void initialize( Properties props ) 452 throws WikiException 453 { 454 m_startTime = new Date(); 455 m_properties = props; 456 457 // 458 // Initialize log4j. However, make sure that we don't initialize it multiple times. 459 // By default we load the log4j config statements from jspwiki.properties, unless 460 // the property jspwiki.use.external.logconfig=true, in that case we let log4j figure out the 461 // logging configuration. 462 // 463 if( !c_configured ) 464 { 465 String useExternalLogConfig = TextUtil.getStringProperty(props,"jspwiki.use.external.logconfig","false"); 466 if( useExternalLogConfig == null || useExternalLogConfig.equals("false")) 467 { 468 PropertyConfigurator.configure( props ); 469 } 470 c_configured = true; 471 } 472 473 log.info("*******************************************"); 474 log.info(Release.APPNAME+" "+Release.getVersionString()+" starting. Whee!"); 475 476 fireEvent( WikiEngineEvent.INITIALIZING ); // begin initialization 477 478 log.debug("Java version: "+System.getProperty("java.runtime.version")); 479 log.debug("Java vendor: "+System.getProperty("java.vm.vendor")); 480 log.debug("OS: "+System.getProperty("os.name")+" "+System.getProperty("os.version")+" "+System.getProperty("os.arch")); 481 log.debug("Default server locale: "+Locale.getDefault()); 482 log.debug("Default server timezone: "+TimeZone.getDefault().getDisplayName(true, TimeZone.LONG)); 483 484 if( m_servletContext != null ) 485 { 486 log.info("Servlet container: "+m_servletContext.getServerInfo() ); 487 if( m_servletContext.getMajorVersion() < 2 || 488 (m_servletContext.getMajorVersion() == 2 && m_servletContext.getMinorVersion() < 4) ) 489 { 490 throw new InternalWikiException("I require a container which supports at least version 2.4 of Servlet specification"); 491 } 492 } 493 494 log.debug("Configuring WikiEngine..."); 495 496 // Initializes the CommandResolver 497 m_commandResolver = new CommandResolver( this, props ); 498 499 // 500 // Create and find the default working directory. 501 // 502 m_workDir = TextUtil.getStringProperty( props, PROP_WORKDIR, null ); 503 504 if( m_workDir == null ) { 505 m_workDir = System.getProperty("java.io.tmpdir", "."); 506 m_workDir += File.separator+Release.APPNAME+"-"+m_appid; 507 } 508 509 try { 510 File f = new File( m_workDir ); 511 f.mkdirs(); 512 513 // 514 // A bunch of sanity checks 515 // 516 if( !f.exists() ) throw new WikiException("Work directory does not exist: "+m_workDir); 517 if( !f.canRead() ) throw new WikiException("No permission to read work directory: "+m_workDir); 518 if( !f.canWrite() ) throw new WikiException("No permission to write to work directory: "+m_workDir); 519 if( !f.isDirectory() ) throw new WikiException("jspwiki.workDir does not point to a directory: "+m_workDir); 520 } catch( SecurityException e ) { 521 log.fatal( "Unable to find or create the working directory: "+m_workDir, e ); 522 throw new IllegalArgumentException( "Unable to find or create the working dir: " + m_workDir, e ); 523 } 524 525 log.info("JSPWiki working directory is '"+m_workDir+"'"); 526 527 m_saveUserInfo = TextUtil.getBooleanProperty( props, PROP_STOREUSERNAME, m_saveUserInfo ); 528 m_useUTF8 = StandardCharsets.UTF_8.name().equals( TextUtil.getStringProperty( props, PROP_ENCODING, StandardCharsets.ISO_8859_1.name() ) ); 529 m_beautifyTitle = TextUtil.getBooleanProperty( props, PROP_BEAUTIFYTITLE, m_beautifyTitle ); 530 m_templateDir = TextUtil.getStringProperty( props, PROP_TEMPLATEDIR, "default" ); 531 enforceValidTemplateDirectory(); 532 m_frontPage = TextUtil.getStringProperty( props, PROP_FRONTPAGE, "Main" ); 533 534 // 535 // Initialize the important modules. Any exception thrown by the 536 // managers means that we will not start up. 537 // 538 539 // FIXME: This part of the code is getting unwieldy. We must think 540 // of a better way to do the startup-sequence. 541 try 542 { 543 Class< ? > urlclass = ClassUtil.findClass( "org.apache.wiki.url", 544 TextUtil.getStringProperty( props, PROP_URLCONSTRUCTOR, "DefaultURLConstructor" ) ); 545 m_urlConstructor = (URLConstructor) urlclass.getDeclaredConstructor().newInstance(); 546 m_urlConstructor.initialize( this, props ); 547 548 m_pageManager = ClassUtil.getMappedObject( PageManager.class.getName(), this, props ); 549 m_pluginManager = ClassUtil.getMappedObject( PluginManager.class.getName(), this, props ); 550 m_differenceManager = ClassUtil.getMappedObject( DifferenceManager.class.getName(), this, props ); 551 m_attachmentManager = ClassUtil.getMappedObject( AttachmentManager.class.getName(), this, props ); 552 m_variableManager = ClassUtil.getMappedObject( VariableManager.class.getName(), props ); 553 m_renderingManager = ClassUtil.getMappedObject( RenderingManager.class.getName() ); 554 m_searchManager = ClassUtil.getMappedObject( SearchManager.class.getName(), this, props ); 555 m_authenticationManager = ClassUtil.getMappedObject( AuthenticationManager.class.getName() ); 556 m_authorizationManager = ClassUtil.getMappedObject( AuthorizationManager.class.getName() ); 557 m_userManager = ClassUtil.getMappedObject( UserManager.class.getName() ); 558 m_groupManager = ClassUtil.getMappedObject( GroupManager.class.getName() ); 559 m_editorManager = ClassUtil.getMappedObject( EditorManager.class.getName(), this ); 560 m_editorManager.initialize( props ); 561 562 m_progressManager = new ProgressManager(); 563 564 // Initialize the authentication, authorization, user and acl managers 565 m_authenticationManager.initialize( this, props ); 566 m_authorizationManager.initialize( this, props ); 567 m_userManager.initialize( this, props ); 568 m_groupManager.initialize( this, props ); 569 m_aclManager = getAclManager(); 570 571 // Start the Workflow manager 572 m_workflowMgr = ClassUtil.getMappedObject(WorkflowManager.class.getName()); 573 m_workflowMgr.initialize(this, props); 574 m_tasksManager = ClassUtil.getMappedObject(TasksManager.class.getName()); 575 576 m_internationalizationManager = ClassUtil.getMappedObject(InternationalizationManager.class.getName(),this); 577 m_templateManager = ClassUtil.getMappedObject(TemplateManager.class.getName(), this, props ); 578 579 // Since we want to use a page filters initilize() method 580 // as a engine startup listener where we can initialize global event listeners, 581 // it must be called lastly, so that all object references in the engine 582 // are availabe to the initialize() method 583 m_filterManager = ClassUtil.getMappedObject(FilterManager.class.getName(), this, props ); 584 585 m_adminBeanManager = ClassUtil.getMappedObject(AdminBeanManager.class.getName(),this); 586 587 // RenderingManager depends on FilterManager events. 588 m_renderingManager.initialize( this, props ); 589 590 // 591 // ReferenceManager has the side effect of loading all 592 // pages. Therefore after this point, all page attributes 593 // are available. 594 // 595 // initReferenceManager is indirectly using m_filterManager, therefore 596 // it has to be called after it was initialized. 597 // 598 initReferenceManager(); 599 600 // 601 // Hook the different manager routines into the system. 602 // 603 m_filterManager.addPageFilter(m_referenceManager, -1001 ); 604 m_filterManager.addPageFilter(m_searchManager, -1002 ); 605 } 606 607 catch( RuntimeException e ) 608 { 609 // RuntimeExceptions may occur here, even if they shouldn't. 610 log.fatal( "Failed to start managers.", e ); 611 throw new WikiException( "Failed to start managers: " + e.getMessage(), e ); 612 } 613 catch (ClassNotFoundException e) 614 { 615 log.fatal( "JSPWiki could not start, URLConstructor was not found: " + e.getMessage(), e ); 616 throw new WikiException(e.getMessage(), e ); 617 } 618 catch (InstantiationException e) 619 { 620 log.fatal( "JSPWiki could not start, URLConstructor could not be instantiated: " + e.getMessage(), e ); 621 throw new WikiException(e.getMessage(), e ); 622 } 623 catch (IllegalAccessException e) 624 { 625 log.fatal( "JSPWiki could not start, URLConstructor cannot be accessed: " + e.getMessage(), e ); 626 throw new WikiException(e.getMessage(), e ); 627 } 628 catch( Exception e ) 629 { 630 // Final catch-all for everything 631 log.fatal( "JSPWiki could not start, due to an unknown exception when starting.",e ); 632 throw new WikiException( "Failed to start. Caused by: " + e.getMessage() + 633 "; please check log files for better information.", e ); 634 } 635 636 // 637 // Initialize the good-to-have-but-not-fatal modules. 638 // 639 try 640 { 641 if( TextUtil.getBooleanProperty( props, 642 RSSGenerator.PROP_GENERATE_RSS, 643 false ) ) 644 { 645 m_rssGenerator = (RSSGenerator)ClassUtil.getMappedObject(RSSGenerator.class.getName(), this, props ); 646 } 647 648 m_pageRenamer = (PageRenamer)ClassUtil.getMappedObject(PageRenamer.class.getName(), this, props ); 649 } 650 catch( Exception e ) 651 { 652 log.error( "Unable to start RSS generator - JSPWiki will still work, "+ 653 "but there will be no RSS feed.", e ); 654 } 655 656 // Start the RSS generator & generator thread 657 if( m_rssGenerator != null ) 658 { 659 m_rssFile = TextUtil.getStringProperty( props, 660 RSSGenerator.PROP_RSSFILE, "rss.rdf" ); 661 File rssFile=null; 662 if (m_rssFile.startsWith(File.separator)) 663 { 664 // honor absolute pathnames: 665 rssFile = new File(m_rssFile ); 666 } 667 else 668 { 669 // relative path names are anchored from the webapp root path: 670 rssFile = new File( getRootPath(), m_rssFile ); 671 } 672 int rssInterval = TextUtil.getIntegerProperty( props, 673 RSSGenerator.PROP_INTERVAL, 3600 ); 674 RSSThread rssThread = new RSSThread( this, rssFile, rssInterval ); 675 rssThread.start(); 676 } 677 678 fireEvent( WikiEngineEvent.INITIALIZED ); // initialization complete 679 680 log.info("WikiEngine configured."); 681 m_isConfigured = true; 682 } 683 684 /** 685 * Checks if the template directory specified in the wiki's properties actually exists. If it doesn't, then {@code m_templateDir} is 686 * set to {@link #DEFAULT_TEMPLATE_NAME}. 687 * <p> 688 * This checks the existence of the <tt>ViewTemplate.jsp</tt> file, which exists in every template using {@code m_servletContext.getRealPath("/")}. 689 * <p> 690 * {@code m_servletContext.getRealPath("/")} can return {@code null} on certain servers/conditions (f.ex, packed wars), an extra check 691 * against {@code m_servletContext.getResource} is made. 692 */ 693 void enforceValidTemplateDirectory() { 694 if( m_servletContext != null ) { 695 final String viewTemplate = "templates" + File.separator + getTemplateDir() + File.separator + "ViewTemplate.jsp"; 696 boolean exists = new File( m_servletContext.getRealPath("/") + viewTemplate ).exists(); 697 if( !exists ) { 698 try { 699 URL url = m_servletContext.getResource( viewTemplate ); 700 exists = url != null && StringUtils.isNotEmpty( url.getFile() ); 701 } catch( MalformedURLException e ) { 702 exists = false; 703 } 704 } 705 if( !exists ) { 706 log.warn( getTemplateDir() + " template not found, updating WikiEngine's default template to " + DEFAULT_TEMPLATE_NAME ); 707 m_templateDir = DEFAULT_TEMPLATE_NAME; 708 } 709 } 710 } 711 712 /** 713 * Initializes the reference manager. Scans all existing WikiPages for 714 * internal links and adds them to the ReferenceManager object. 715 * 716 * @throws WikiException If the reference manager initialization fails. 717 */ 718 public void initReferenceManager() throws WikiException { 719 try { 720 ArrayList<WikiPage> pages = new ArrayList<>(); 721 pages.addAll( m_pageManager.getAllPages() ); 722 pages.addAll( m_attachmentManager.getAllAttachments() ); 723 724 // Build a new manager with default key lists. 725 if( m_referenceManager == null ) 726 { 727 m_referenceManager = ClassUtil.getMappedObject(ReferenceManager.class.getName(), this ); 728 m_referenceManager.initialize( pages ); 729 } 730 731 } catch( ProviderException e ) { 732 log.fatal("PageProvider is unable to list pages: ", e); 733 } catch( ReflectiveOperationException | IllegalArgumentException e ) { 734 throw new WikiException( "Could not instantiate ReferenceManager: " + e.getMessage(), e ); 735 } 736 } 737 738 /** 739 * Returns the set of properties that the WikiEngine was initialized 740 * with. Note that this method returns a direct reference, so it's possible 741 * to manipulate the properties. However, this is not advised unless you 742 * really know what you're doing. 743 * 744 * @return The wiki properties 745 */ 746 747 public Properties getWikiProperties() 748 { 749 return m_properties; 750 } 751 752 /** 753 * Returns the JSPWiki working directory set with "jspwiki.workDir". 754 * 755 * @since 2.1.100 756 * @return The working directory. 757 */ 758 public String getWorkDir() 759 { 760 return m_workDir; 761 } 762 763 /** 764 * Returns the current template directory. 765 * 766 * @since 1.9.20 767 * @return The template directory as initialized by the engine. 768 */ 769 public String getTemplateDir() 770 { 771 return m_templateDir; 772 } 773 774 /** 775 * Returns the current TemplateManager. 776 * 777 * @return A TemplateManager instance. 778 */ 779 public TemplateManager getTemplateManager() 780 { 781 return m_templateManager; 782 } 783 784 /** 785 * Returns the base URL, telling where this Wiki actually lives. 786 * 787 * @since 1.6.1 788 * @return The Base URL. 789 */ 790 public String getBaseURL() { 791 return m_servletContext.getContextPath(); 792 } 793 794 /** 795 * Returns the moment when this engine was started. 796 * 797 * @since 2.0.15. 798 * @return The start time of this wiki. 799 */ 800 public Date getStartTime() 801 { 802 return (Date)m_startTime.clone(); 803 } 804 805 /** 806 * <p> 807 * Returns the basic absolute URL to a page, without any modifications. You 808 * may add any parameters to this. 809 * </p> 810 * <p> 811 * Since 2.3.90 it is safe to call this method with <code>null</code> 812 * pageName, in which case it will default to the front page. 813 * </p> 814 * @since 2.0.3 815 * @param pageName The name of the page. May be null, in which case defaults to the front page. 816 * @return An absolute URL to the page. 817 */ 818 public String getViewURL( String pageName ) { 819 if( pageName == null ) { 820 pageName = getFrontPage(); 821 } 822 return getURLConstructor().makeURL( WikiContext.VIEW, pageName, "absolute".equals( PROP_REFSTYLE ), null ); 823 } 824 825 /** 826 * Returns an URL if a WikiContext is not available. 827 * 828 * @param context The WikiContext (VIEW, EDIT, etc...) 829 * @param pageName Name of the page, as usual 830 * @param params List of parameters. May be null, if no parameters. 831 * @param absolute If true, will generate an absolute URL regardless of properties setting. 832 * @return An URL (absolute or relative). 833 */ 834 public String getURL( final String context, String pageName, final String params, final boolean absolute ) { 835 if( pageName == null ) { 836 pageName = getFrontPage(); 837 } 838 return m_urlConstructor.makeURL( context, pageName, absolute, params ); 839 } 840 841 /** 842 * Returns the default front page, if no page is used. 843 * 844 * @return The front page name. 845 */ 846 847 public String getFrontPage() 848 { 849 return m_frontPage; 850 } 851 852 /** 853 * Returns the ServletContext that this particular WikiEngine was 854 * initialized with. <B>It may return null</B>, if the WikiEngine is not 855 * running inside a servlet container! 856 * 857 * @since 1.7.10 858 * @return ServletContext of the WikiEngine, or null. 859 */ 860 861 public ServletContext getServletContext() 862 { 863 return m_servletContext; 864 } 865 866 /** 867 * Returns the query string (the portion after the question mark). 868 * 869 * @param request The HTTP request to parse. 870 * @return The query string. If the query string is null, 871 * returns an empty string. 872 * 873 * @since 2.1.3 874 */ 875 public String safeGetQueryString( final HttpServletRequest request ) { 876 if (request == null) { 877 return ""; 878 } 879 880 String res = request.getQueryString(); 881 if( res != null ) { 882 res = new String( res.getBytes( StandardCharsets.ISO_8859_1 ), getContentEncoding() ); 883 884 // 885 // Ensure that the 'page=xyz' attribute is removed 886 // FIXME: Is it really the mandate of this routine to do that? 887 // 888 final int pos1 = res.indexOf("page="); 889 if( pos1 >= 0 ) { 890 String tmpRes = res.substring( 0, pos1 ); 891 final int pos2 = res.indexOf( "&",pos1 ) + 1; 892 if ( ( pos2 > 0 ) && ( pos2 < res.length() ) ) { 893 tmpRes = tmpRes + res.substring(pos2); 894 } 895 res = tmpRes; 896 } 897 } 898 899 return res; 900 } 901 902 /** 903 * Returns an URL to some other Wiki that we know. 904 * 905 * @param wikiName The name of the other wiki. 906 * @return null, if no such reference was found. 907 */ 908 public String getInterWikiURL( String wikiName ) 909 { 910 return TextUtil.getStringProperty(m_properties,PROP_INTERWIKIREF+wikiName,null); 911 } 912 913 /** 914 * Returns a collection of all supported InterWiki links. 915 * 916 * @return A Collection of Strings. 917 */ 918 public Collection< String > getAllInterWikiLinks() 919 { 920 ArrayList< String > list = new ArrayList< >(); 921 922 for( Enumeration< ? > i = m_properties.propertyNames(); i.hasMoreElements(); ) 923 { 924 String prop = ( String )i.nextElement(); 925 926 if( prop.startsWith( PROP_INTERWIKIREF ) ) 927 { 928 list.add( prop.substring( prop.lastIndexOf( "." ) + 1 ) ); 929 } 930 } 931 932 return list; 933 } 934 935 /** 936 * Returns a collection of all image types that get inlined. 937 * 938 * @return A Collection of Strings with a regexp pattern. 939 */ 940 public Collection< String > getAllInlinedImagePatterns() 941 { 942 Properties props = getWikiProperties(); 943 ArrayList<String> ptrnlist = new ArrayList<>(); 944 945 for( Enumeration< ? > e = props.propertyNames(); e.hasMoreElements(); ) 946 { 947 String name = ( String )e.nextElement(); 948 949 if( name.startsWith( PROP_INLINEIMAGEPTRN ) ) 950 { 951 String ptrn = TextUtil.getStringProperty( props, name, null ); 952 953 ptrnlist.add( ptrn ); 954 } 955 } 956 957 if( ptrnlist.size() == 0 ) 958 { 959 ptrnlist.add( DEFAULT_INLINEPATTERN ); 960 } 961 962 return ptrnlist; 963 } 964 965 /** 966 * <p>If the page is a special page, then returns a direct URL 967 * to that page. Otherwise returns <code>null</code>. 968 * This method delegates requests to 969 * {@link org.apache.wiki.ui.CommandResolver#getSpecialPageReference(String)}. 970 * </p> 971 * <p> 972 * Special pages are defined in jspwiki.properties using the jspwiki.specialPage 973 * setting. They're typically used to give Wiki page names to e.g. custom JSP 974 * pages. 975 * </p> 976 * 977 * @param original The page to check 978 * @return A reference to the page, or null, if there's no special page. 979 */ 980 public String getSpecialPageReference( String original ) 981 { 982 return m_commandResolver.getSpecialPageReference( original ); 983 } 984 985 /** 986 * Returns the name of the application. 987 * 988 * @return A string describing the name of this application. 989 */ 990 991 // FIXME: Should use servlet context as a default instead of a constant. 992 public String getApplicationName() 993 { 994 String appName = TextUtil.getStringProperty(m_properties,PROP_APPNAME,Release.APPNAME); 995 996 return MarkupParser.cleanLink( appName ); 997 } 998 999 /** 1000 * Beautifies the title of the page by appending spaces in suitable 1001 * places, if the user has so decreed in the properties when constructing 1002 * this WikiEngine. However, attachment names are only beautified by 1003 * the name. 1004 * 1005 * @param title The title to beautify 1006 * @return A beautified title (or, if beautification is off, 1007 * returns the title without modification) 1008 * @since 1.7.11 1009 */ 1010 public String beautifyTitle( String title ) 1011 { 1012 if( m_beautifyTitle ) 1013 { 1014 try 1015 { 1016 Attachment att = m_attachmentManager.getAttachmentInfo(title); 1017 1018 if(att == null) 1019 { 1020 return TextUtil.beautifyString( title ); 1021 } 1022 1023 String parent = TextUtil.beautifyString( att.getParentName() ); 1024 1025 return parent + "/" + att.getFileName(); 1026 } 1027 catch( ProviderException e ) 1028 { 1029 return title; 1030 } 1031 } 1032 1033 return title; 1034 } 1035 1036 /** 1037 * Beautifies the title of the page by appending non-breaking spaces 1038 * in suitable places. This is really suitable only for HTML output, 1039 * as it uses the &nbsp; -character. 1040 * 1041 * @param title The title to beautify 1042 * @return A beautified title. 1043 * @since 2.1.127 1044 */ 1045 public String beautifyTitleNoBreak( String title ) 1046 { 1047 if( m_beautifyTitle ) 1048 { 1049 return TextUtil.beautifyString( title, " " ); 1050 } 1051 1052 return title; 1053 } 1054 1055 /** 1056 * Returns true, if the requested page (or an alias) exists. Will consider 1057 * any version as existing. Will also consider attachments. 1058 * 1059 * @param page WikiName of the page. 1060 * @return true, if page (or attachment) exists. 1061 */ 1062 public boolean pageExists( String page ) 1063 { 1064 Attachment att = null; 1065 1066 try 1067 { 1068 if( m_commandResolver.getSpecialPageReference(page) != null ) return true; 1069 1070 if( getFinalPageName( page ) != null ) 1071 { 1072 return true; 1073 } 1074 1075 att = getAttachmentManager().getAttachmentInfo( (WikiContext)null, page ); 1076 } 1077 catch( ProviderException e ) 1078 { 1079 log.debug("pageExists() failed to find attachments",e); 1080 } 1081 1082 return att != null; 1083 } 1084 1085 /** 1086 * Returns true, if the requested page (or an alias) exists with the 1087 * requested version. 1088 * 1089 * @param page Page name 1090 * @param version Page version 1091 * @return True, if page (or alias, or attachment) exists 1092 * @throws ProviderException If the provider fails. 1093 */ 1094 public boolean pageExists( String page, int version ) 1095 throws ProviderException 1096 { 1097 if( m_commandResolver.getSpecialPageReference(page) != null ) return true; 1098 1099 String finalName = getFinalPageName( page ); 1100 1101 boolean isThere = false; 1102 1103 if( finalName != null ) 1104 { 1105 // 1106 // Go and check if this particular version of this page 1107 // exists. 1108 // 1109 isThere = m_pageManager.pageExists( finalName, version ); 1110 } 1111 1112 if( isThere == false ) 1113 { 1114 // 1115 // Go check if such an attachment exists. 1116 // 1117 try 1118 { 1119 isThere = getAttachmentManager().getAttachmentInfo( (WikiContext)null, page, version ) != null; 1120 } 1121 catch( ProviderException e ) 1122 { 1123 log.debug("pageExists() failed to find attachments",e); 1124 } 1125 } 1126 1127 return isThere; 1128 } 1129 1130 /** 1131 * Returns true, if the requested page (or an alias) exists, with the 1132 * specified version in the WikiPage. 1133 * 1134 * @param page A WikiPage object describing the name and version. 1135 * @return true, if the page (or alias, or attachment) exists. 1136 * @throws ProviderException If something goes badly wrong. 1137 * @since 2.0 1138 */ 1139 public boolean pageExists( WikiPage page ) 1140 throws ProviderException 1141 { 1142 if( page != null ) 1143 { 1144 return pageExists( page.getName(), page.getVersion() ); 1145 } 1146 return false; 1147 } 1148 1149 /** 1150 * Returns the correct page name, or null, if no such 1151 * page can be found. Aliases are considered. This 1152 * method simply delegates to 1153 * {@link org.apache.wiki.ui.CommandResolver#getFinalPageName(String)}. 1154 * @since 2.0 1155 * @param page Page name. 1156 * @return The rewritten page name, or null, if the page does not exist. 1157 * @throws ProviderException If something goes wrong in the backend. 1158 */ 1159 public String getFinalPageName( String page ) 1160 throws ProviderException 1161 { 1162 return m_commandResolver.getFinalPageName( page ); 1163 } 1164 1165 /** 1166 * Turns a WikiName into something that can be 1167 * called through using an URL. 1168 * 1169 * @since 1.4.1 1170 * @param pagename A name. Can be actually any string. 1171 * @return A properly encoded name. 1172 * @see #decodeName(String) 1173 */ 1174 public String encodeName( String pagename ) 1175 { 1176 try 1177 { 1178 return URLEncoder.encode( pagename, m_useUTF8 ? "UTF-8" : "ISO-8859-1" ); 1179 } 1180 catch( UnsupportedEncodingException e ) 1181 { 1182 throw new InternalWikiException( "ISO-8859-1 not a supported encoding!?! Your platform is borked." , e); 1183 } 1184 } 1185 1186 /** 1187 * Decodes a URL-encoded request back to regular life. This properly heeds 1188 * the encoding as defined in the settings file. 1189 * 1190 * @param pagerequest The URL-encoded string to decode 1191 * @return A decoded string. 1192 * @see #encodeName(String) 1193 */ 1194 public String decodeName( final String pagerequest ) { 1195 try { 1196 return URLDecoder.decode( pagerequest, m_useUTF8 ? "UTF-8" : "ISO-8859-1" ); 1197 } catch( UnsupportedEncodingException e ) { 1198 throw new InternalWikiException("ISO-8859-1 not a supported encoding!?! Your platform is borked.", e); 1199 } 1200 } 1201 1202 /** 1203 * Returns the IANA name of the character set encoding we're 1204 * supposed to be using right now. 1205 * 1206 * @since 1.5.3 1207 * @return The content encoding (either UTF-8 or ISO-8859-1). 1208 */ 1209 public Charset getContentEncoding() { 1210 if( m_useUTF8 ) { 1211 return StandardCharsets.UTF_8; 1212 } 1213 return StandardCharsets.ISO_8859_1; 1214 } 1215 1216 /** 1217 * Returns the {@link org.apache.wiki.workflow.WorkflowManager} associated with this 1218 * WikiEngine. If the WIkiEngine has not been initialized, this method will return 1219 * <code>null</code>. 1220 * @return the task queue 1221 */ 1222 public WorkflowManager getWorkflowManager() 1223 { 1224 return m_workflowMgr; 1225 } 1226 1227 /** 1228 * Returns the un-HTMLized text of the latest version of a page. 1229 * This method also replaces the < and & -characters with 1230 * their respective HTML entities, thus making it suitable 1231 * for inclusion on an HTML page. If you want to have the 1232 * page text without any conversions, use getPureText(). 1233 * 1234 * @param page WikiName of the page to fetch. 1235 * @return WikiText. 1236 */ 1237 public String getText( String page ) 1238 { 1239 return getText( page, WikiPageProvider.LATEST_VERSION ); 1240 } 1241 1242 /** 1243 * Returns the un-HTMLized text of the given version of a page. 1244 * This method also replaces the < and & -characters with 1245 * their respective HTML entities, thus making it suitable 1246 * for inclusion on an HTML page. If you want to have the 1247 * page text without any conversions, use getPureText(). 1248 * 1249 * 1250 * @param page WikiName of the page to fetch 1251 * @param version Version of the page to fetch 1252 * @return WikiText. 1253 */ 1254 public String getText( String page, int version ) 1255 { 1256 String result = getPureText( page, version ); 1257 1258 // 1259 // Replace ampersand first, or else all quotes and stuff 1260 // get replaced as well with " etc. 1261 // 1262 /* 1263 result = TextUtil.replaceString( result, "&", "&" ); 1264 */ 1265 1266 result = TextUtil.replaceEntities( result ); 1267 1268 return result; 1269 } 1270 1271 /** 1272 * Returns the un-HTMLized text of the given version of a page in 1273 * the given context. USE THIS METHOD if you don't know what 1274 * doing. 1275 * <p> 1276 * This method also replaces the < and & -characters with 1277 * their respective HTML entities, thus making it suitable 1278 * for inclusion on an HTML page. If you want to have the 1279 * page text without any conversions, use getPureText(). 1280 * 1281 * @since 1.9.15. 1282 * @param context The WikiContext 1283 * @param page A page reference (not an attachment) 1284 * @return The page content as HTMLized String. 1285 * @see #getPureText(WikiPage) 1286 */ 1287 public String getText( WikiContext context, WikiPage page ) 1288 { 1289 return getText( page.getName(), page.getVersion() ); 1290 } 1291 1292 1293 /** 1294 * Returns the pure text of a page, no conversions. Use this 1295 * if you are writing something that depends on the parsing 1296 * of the page. Note that you should always check for page 1297 * existence through pageExists() before attempting to fetch 1298 * the page contents. 1299 * 1300 * @param page The name of the page to fetch. 1301 * @param version If WikiPageProvider.LATEST_VERSION, then uses the 1302 * latest version. 1303 * @return The page contents. If the page does not exist, 1304 * returns an empty string. 1305 */ 1306 // FIXME: Should throw an exception on unknown page/version? 1307 public String getPureText( String page, int version ) 1308 { 1309 String result = null; 1310 1311 try 1312 { 1313 result = m_pageManager.getPageText( page, version ); 1314 } 1315 catch( ProviderException e ) 1316 { 1317 // FIXME 1318 } 1319 finally 1320 { 1321 if( result == null ) 1322 result = ""; 1323 } 1324 1325 return result; 1326 } 1327 1328 /** 1329 * Returns the pure text of a page, no conversions. Use this 1330 * if you are writing something that depends on the parsing 1331 * the page. Note that you should always check for page 1332 * existence through pageExists() before attempting to fetch 1333 * the page contents. 1334 * 1335 * @param page A handle to the WikiPage 1336 * @return String of WikiText. 1337 * @since 2.1.13. 1338 */ 1339 public String getPureText( WikiPage page ) 1340 { 1341 return getPureText( page.getName(), page.getVersion() ); 1342 } 1343 1344 /** 1345 * Returns the converted HTML of the page using a different 1346 * context than the default context. 1347 * 1348 * @param context A WikiContext in which you wish to render this page in. 1349 * @param page WikiPage reference. 1350 * @return HTML-rendered version of the page. 1351 */ 1352 1353 public String getHTML( WikiContext context, WikiPage page ) 1354 { 1355 String pagedata = null; 1356 1357 pagedata = getPureText( page.getName(), page.getVersion() ); 1358 1359 String res = textToHTML( context, pagedata ); 1360 1361 return res; 1362 } 1363 1364 /** 1365 * Returns the converted HTML of the page. 1366 * 1367 * @param page WikiName of the page to convert. 1368 * @return HTML-rendered version of the page. 1369 */ 1370 public String getHTML( String page ) 1371 { 1372 return getHTML( page, WikiPageProvider.LATEST_VERSION ); 1373 } 1374 1375 /** 1376 * Returns the converted HTML of the page's specific version. 1377 * The version must be a positive integer, otherwise the current 1378 * version is returned. 1379 * 1380 * @param pagename WikiName of the page to convert. 1381 * @param version Version number to fetch 1382 * @return HTML-rendered page text. 1383 */ 1384 public String getHTML( String pagename, int version ) 1385 { 1386 WikiPage page = getPage( pagename, version ); 1387 1388 WikiContext context = new WikiContext( this, 1389 page ); 1390 context.setRequestContext( WikiContext.NONE ); 1391 1392 String res = getHTML( context, page ); 1393 1394 return res; 1395 } 1396 1397 /** 1398 * Converts raw page data to HTML. 1399 * 1400 * @param pagedata Raw page data to convert to HTML 1401 * @param context The WikiContext in which the page is to be rendered 1402 * @return Rendered page text 1403 */ 1404 public String textToHTML( WikiContext context, String pagedata ) 1405 { 1406 String result = ""; 1407 1408 boolean runFilters = "true".equals(m_variableManager.getValue(context,PROP_RUNFILTERS,"true")); 1409 1410 StopWatch sw = new StopWatch(); 1411 sw.start(); 1412 try 1413 { 1414 if( runFilters ) 1415 pagedata = m_filterManager.doPreTranslateFiltering( context, pagedata ); 1416 1417 result = m_renderingManager.getHTML( context, pagedata ); 1418 1419 if( runFilters ) 1420 result = m_filterManager.doPostTranslateFiltering( context, result ); 1421 } 1422 catch( FilterException e ) 1423 { 1424 // FIXME: Don't yet know what to do 1425 } 1426 sw.stop(); 1427 if( log.isDebugEnabled() ) 1428 log.debug("Page "+context.getRealPage().getName()+" rendered, took "+sw ); 1429 1430 return result; 1431 } 1432 1433 /** 1434 * Protected method that signals that the WikiEngine will be 1435 * shut down by the servlet container. It is called by 1436 * {@link WikiServlet#destroy()}. When this method is called, 1437 * it fires a "shutdown" WikiEngineEvent to all registered 1438 * listeners. 1439 */ 1440 protected void shutdown() 1441 { 1442 fireEvent( WikiEngineEvent.SHUTDOWN ); 1443 m_filterManager.destroy(); 1444 } 1445 1446 /** 1447 * Reads a WikiPageful of data from a String and returns all links 1448 * internal to this Wiki in a Collection. 1449 * 1450 * @param page The WikiPage to scan 1451 * @param pagedata The page contents 1452 * @return a Collection of Strings 1453 */ 1454 public Collection< String > scanWikiLinks( WikiPage page, String pagedata ) { 1455 LinkCollector localCollector = new LinkCollector(); 1456 1457 textToHTML( new WikiContext( this, page ), 1458 pagedata, 1459 localCollector, 1460 null, 1461 localCollector, 1462 false, 1463 true ); 1464 1465 return localCollector.getLinks(); 1466 } 1467 1468 /** 1469 * Just convert WikiText to HTML. 1470 * 1471 * @param context The WikiContext in which to do the conversion 1472 * @param pagedata The data to render 1473 * @param localLinkHook Is called whenever a wiki link is found 1474 * @param extLinkHook Is called whenever an external link is found 1475 * 1476 * @return HTML-rendered page text. 1477 */ 1478 1479 public String textToHTML( WikiContext context, 1480 String pagedata, 1481 StringTransmutator localLinkHook, 1482 StringTransmutator extLinkHook ) 1483 { 1484 return textToHTML( context, pagedata, localLinkHook, extLinkHook, null, true, false ); 1485 } 1486 1487 /** 1488 * Just convert WikiText to HTML. 1489 * 1490 * @param context The WikiContext in which to do the conversion 1491 * @param pagedata The data to render 1492 * @param localLinkHook Is called whenever a wiki link is found 1493 * @param extLinkHook Is called whenever an external link is found 1494 * @param attLinkHook Is called whenever an attachment link is found 1495 * @return HTML-rendered page text. 1496 */ 1497 1498 public String textToHTML( WikiContext context, 1499 String pagedata, 1500 StringTransmutator localLinkHook, 1501 StringTransmutator extLinkHook, 1502 StringTransmutator attLinkHook ) 1503 { 1504 return textToHTML( context, pagedata, localLinkHook, extLinkHook, attLinkHook, true, false ); 1505 } 1506 1507 /** 1508 * Helper method for doing the HTML translation. 1509 * 1510 * @param context The WikiContext in which to do the conversion 1511 * @param pagedata The data to render 1512 * @param localLinkHook Is called whenever a wiki link is found 1513 * @param extLinkHook Is called whenever an external link is found 1514 * @param parseAccessRules Parse the access rules if we encounter them 1515 * @param justParse Just parses the pagedata, does not actually render. In this case, 1516 * this methods an empty string. 1517 * @return HTML-rendered page text. 1518 1519 */ 1520 private String textToHTML( WikiContext context, 1521 String pagedata, 1522 StringTransmutator localLinkHook, 1523 StringTransmutator extLinkHook, 1524 StringTransmutator attLinkHook, 1525 boolean parseAccessRules, 1526 boolean justParse ) 1527 { 1528 String result = ""; 1529 1530 if( pagedata == null ) 1531 { 1532 log.error("NULL pagedata to textToHTML()"); 1533 return null; 1534 } 1535 1536 boolean runFilters = "true".equals(m_variableManager.getValue(context,PROP_RUNFILTERS,"true")); 1537 1538 try 1539 { 1540 StopWatch sw = new StopWatch(); 1541 sw.start(); 1542 1543 if( runFilters && m_filterManager != null ) 1544 pagedata = m_filterManager.doPreTranslateFiltering( context, pagedata ); 1545 1546 MarkupParser mp = m_renderingManager.getParser( context, pagedata ); 1547 mp.addLocalLinkHook( localLinkHook ); 1548 mp.addExternalLinkHook( extLinkHook ); 1549 mp.addAttachmentLinkHook( attLinkHook ); 1550 1551 if( !parseAccessRules ) mp.disableAccessRules(); 1552 1553 WikiDocument doc = mp.parse(); 1554 1555 // 1556 // In some cases it's better just to parse, not to render 1557 // 1558 if( !justParse ) 1559 { 1560 result = m_renderingManager.getHTML( context, doc ); 1561 1562 if( runFilters && m_filterManager != null ) 1563 result = m_filterManager.doPostTranslateFiltering( context, result ); 1564 } 1565 1566 sw.stop(); 1567 1568 if( log.isDebugEnabled() ) 1569 log.debug("Page "+context.getRealPage().getName()+" rendered, took "+sw ); 1570 } 1571 catch( IOException e ) 1572 { 1573 log.error( "Failed to scan page data: ", e ); 1574 } 1575 catch( FilterException e ) 1576 { 1577 log.error( "page filter threw exception: ", e ); 1578 // FIXME: Don't yet know what to do 1579 } 1580 1581 return result; 1582 } 1583 1584 /** 1585 * Updates all references for the given page. 1586 * 1587 * @param page wiki page for which references should be updated 1588 */ 1589 public void updateReferences( WikiPage page ) 1590 { 1591 String pageData = getPureText( page.getName(), WikiProvider.LATEST_VERSION ); 1592 1593 m_referenceManager.updateReferences( page.getName(), 1594 scanWikiLinks( page, pageData ) ); 1595 } 1596 1597 1598 /** 1599 * Writes the WikiText of a page into the page repository. If the <code>jspwiki.properties</code> file contains 1600 * the property <code>jspwiki.approver.workflow.saveWikiPage</code> and its value resolves to a valid user, 1601 * {@link org.apache.wiki.auth.authorize.Group} or {@link org.apache.wiki.auth.authorize.Role}, this method will 1602 * place a {@link org.apache.wiki.workflow.Decision} in the approver's workflow inbox and throw a 1603 * {@link org.apache.wiki.workflow.DecisionRequiredException}. If the submitting user is authenticated and the 1604 * page save is rejected, a notification will be placed in the user's decision queue. 1605 * 1606 * @since 2.1.28 1607 * @param context The current WikiContext 1608 * @param text The Wiki markup for the page. 1609 * @throws WikiException if the save operation encounters an error during the save operation. If the page-save 1610 * operation requires approval, the exception will be of type {@link org.apache.wiki.workflow.DecisionRequiredException}. 1611 * Individual PageFilters, such as the {@link org.apache.wiki.filters.SpamFilter} may also throw a 1612 * {@link org.apache.wiki.api.exceptions.RedirectException}. 1613 */ 1614 public void saveText( WikiContext context, String text ) throws WikiException { 1615 // Check if page data actually changed; bail if not 1616 WikiPage page = context.getPage(); 1617 String oldText = getPureText( page ); 1618 String proposedText = TextUtil.normalizePostData( text ); 1619 if ( oldText != null && oldText.equals( proposedText ) ) { 1620 return; 1621 } 1622 1623 // Check if creation of empty pages is allowed; bail if not 1624 boolean allowEmpty = TextUtil.getBooleanProperty( m_properties, PROP_ALLOW_CREATION_OF_EMPTY_PAGES, false ); 1625 if ( !allowEmpty && !pageExists( page ) && text.trim().equals( "" ) ) { 1626 return; 1627 } 1628 1629 // Create approval workflow for page save; add the diffed, proposed and old text versions as 1630 // Facts for the approver (if approval is required). If submitter is authenticated, any reject 1631 // messages will appear in his/her workflow inbox. 1632 WorkflowBuilder builder = WorkflowBuilder.getBuilder( this ); 1633 Principal submitter = context.getCurrentUser(); 1634 Step prepTask = m_tasksManager.buildPreSaveWikiPageTask( context, proposedText ); 1635 Step completionTask = m_tasksManager.buildSaveWikiPageTask(); 1636 String diffText = m_differenceManager.makeDiff( context, oldText, proposedText ); 1637 boolean isAuthenticated = context.getWikiSession().isAuthenticated(); 1638 Fact[] facts = new Fact[ 5 ]; 1639 facts[ 0 ] = new Fact( WorkflowManager.WF_WP_SAVE_FACT_PAGE_NAME, page.getName() ); 1640 facts[ 1 ] = new Fact( WorkflowManager.WF_WP_SAVE_FACT_DIFF_TEXT, diffText ); 1641 facts[ 2 ] = new Fact( WorkflowManager.WF_WP_SAVE_FACT_PROPOSED_TEXT, proposedText ); 1642 facts[ 3 ] = new Fact( WorkflowManager.WF_WP_SAVE_FACT_CURRENT_TEXT, oldText); 1643 facts[ 4 ] = new Fact( WorkflowManager.WF_WP_SAVE_FACT_IS_AUTHENTICATED, Boolean.valueOf( isAuthenticated ) ); 1644 String rejectKey = isAuthenticated ? WorkflowManager.WF_WP_SAVE_REJECT_MESSAGE_KEY : null; 1645 Workflow workflow = builder.buildApprovalWorkflow( submitter, 1646 WorkflowManager.WF_WP_SAVE_APPROVER, 1647 prepTask, 1648 WorkflowManager.WF_WP_SAVE_DECISION_MESSAGE_KEY, 1649 facts, 1650 completionTask, 1651 rejectKey ); 1652 m_workflowMgr.start( workflow ); 1653 1654 // Let callers know if the page-save requires approval 1655 if ( workflow.getCurrentStep() instanceof Decision ) { 1656 throw new DecisionRequiredException( "The page contents must be approved before they become active." ); 1657 } 1658 } 1659 1660 /** 1661 * Returns the number of pages in this Wiki 1662 * @return The total number of pages. 1663 */ 1664 public int getPageCount() 1665 { 1666 return m_pageManager.getTotalPageCount(); 1667 } 1668 1669 /** 1670 * Returns the provider name. 1671 * @return The full class name of the current page provider. 1672 */ 1673 1674 public String getCurrentProvider() 1675 { 1676 return m_pageManager.getProvider().getClass().getName(); 1677 } 1678 1679 /** 1680 * Return information about current provider. This method just calls 1681 * the corresponding PageManager method, which in turn calls the 1682 * provider method. 1683 * 1684 * @return A textual description of the current provider. 1685 * @since 1.6.4 1686 */ 1687 public String getCurrentProviderInfo() 1688 { 1689 return m_pageManager.getProviderDescription(); 1690 } 1691 1692 /** 1693 * Returns a Collection of WikiPages, sorted in time 1694 * order of last change (i.e. first object is the most 1695 * recently changed). This method also includes attachments. 1696 * 1697 * @return Set of WikiPage objects. 1698 */ 1699 1700 // FIXME: Should really get a Date object and do proper comparisons. This is terribly wasteful. 1701 public Set< WikiPage > getRecentChanges() 1702 { 1703 try { 1704 Collection<WikiPage> pages = m_pageManager.getAllPages(); 1705 Collection<Attachment> atts = m_attachmentManager.getAllAttachments(); 1706 1707 TreeSet<WikiPage> sortedPages = new TreeSet<>( new PageTimeComparator() ); 1708 1709 sortedPages.addAll( pages ); 1710 sortedPages.addAll( atts ); 1711 1712 return sortedPages; 1713 } catch( ProviderException e ) { 1714 log.error( "Unable to fetch all pages: ",e); 1715 return null; 1716 } 1717 } 1718 1719 /** 1720 * Finds the corresponding WikiPage object based on the page name. It always finds 1721 * the latest version of a page. 1722 * 1723 * @param pagereq The name of the page to look for. 1724 * @return A WikiPage object, or null, if the page by the name could not be found. 1725 */ 1726 1727 public WikiPage getPage( String pagereq ) 1728 { 1729 return getPage( pagereq, WikiProvider.LATEST_VERSION ); 1730 } 1731 1732 /** 1733 * Finds the corresponding WikiPage object base on the page name and version. 1734 * 1735 * @param pagereq The name of the page to look for. 1736 * @param version The version number to look for. May be WikiProvider.LATEST_VERSION, 1737 * in which case it will look for the latest version (and this method then becomes 1738 * the equivalent of getPage(String). 1739 * 1740 * @return A WikiPage object, or null, if the page could not be found; or if there 1741 * is no such version of the page. 1742 * @since 1.6.7. 1743 */ 1744 1745 public WikiPage getPage( String pagereq, int version ) 1746 { 1747 try 1748 { 1749 WikiPage p = m_pageManager.getPageInfo( pagereq, version ); 1750 1751 if( p == null ) 1752 { 1753 p = m_attachmentManager.getAttachmentInfo( (WikiContext)null, pagereq ); 1754 } 1755 1756 return p; 1757 } 1758 catch( ProviderException e ) 1759 { 1760 log.error( "Unable to fetch page info",e); 1761 return null; 1762 } 1763 } 1764 1765 1766 /** 1767 * Returns a Collection of WikiPages containing the version history of a page. 1768 * 1769 * @param page Name of the page to look for 1770 * @return an ordered List of WikiPages, each corresponding to a different revision of the page. 1771 */ 1772 public List< ? extends WikiPage > getVersionHistory( String page ) { 1773 List< ? extends WikiPage > c = null; 1774 1775 try { 1776 c = m_pageManager.getVersionHistory( page ); 1777 1778 if( c == null ) { 1779 c = m_attachmentManager.getVersionHistory( page ); 1780 } 1781 } catch( ProviderException e ) { 1782 log.error( "FIXME", e ); 1783 } 1784 1785 return c; 1786 } 1787 1788 /** 1789 * Returns a diff of two versions of a page. 1790 * <p> 1791 * Note that the API was changed in 2.6 to provide a WikiContext object! 1792 * 1793 * @param context The WikiContext of the page you wish to get a diff from 1794 * @param version1 Version number of the old page. If WikiPageProvider.LATEST_VERSION (-1), then uses current page. 1795 * @param version2 Version number of the new page. If WikiPageProvider.LATEST_VERSION (-1), then uses current page. 1796 * 1797 * @return A HTML-ized difference between two pages. If there is no difference, returns an empty string. 1798 */ 1799 public String getDiff( WikiContext context, int version1, int version2 ) 1800 { 1801 String page = context.getPage().getName(); 1802 String page1 = getPureText( page, version1 ); 1803 String page2 = getPureText( page, version2 ); 1804 1805 // Kludge to make diffs for new pages to work this way. 1806 1807 if( version1 == WikiPageProvider.LATEST_VERSION ) 1808 { 1809 page1 = ""; 1810 } 1811 1812 String diff = m_differenceManager.makeDiff( context, page1, page2 ); 1813 1814 return diff; 1815 } 1816 1817 /** 1818 * Returns this object's ReferenceManager. 1819 * @return The current ReferenceManager instance. 1820 * 1821 * @since 1.6.1 1822 */ 1823 public ReferenceManager getReferenceManager() 1824 { 1825 return m_referenceManager; 1826 } 1827 1828 /** 1829 * Returns the current rendering manager for this wiki application. 1830 * 1831 * @since 2.3.27 1832 * @return A RenderingManager object. 1833 */ 1834 public RenderingManager getRenderingManager() 1835 { 1836 return m_renderingManager; 1837 } 1838 1839 /** 1840 * Returns the current plugin manager. 1841 * 1842 * In 2.10 the PluginManager will be returned instead of the generic 1843 * 1844 * @since 1.6.1 1845 * @return The current PluginManager instance 1846 */ 1847 @SuppressWarnings("unchecked") 1848 public < T extends PluginManager > T getPluginManager() 1849 { 1850 return (T)m_pluginManager; 1851 } 1852 1853 /** 1854 * Returns the current variable manager. 1855 * @return The current VariableManager. 1856 */ 1857 1858 public VariableManager getVariableManager() 1859 { 1860 return m_variableManager; 1861 } 1862 1863 /** 1864 * Shortcut to getVariableManager().getValue(). However, this method does not 1865 * throw a NoSuchVariableException, but returns null in case the variable does 1866 * not exist. 1867 * 1868 * @param context WikiContext to look the variable in 1869 * @param name Name of the variable to look for 1870 * @return Variable value, or null, if there is no such variable. 1871 * @since 2.2 1872 */ 1873 public String getVariable( WikiContext context, String name ) { 1874 try { 1875 return m_variableManager.getValue( context, name ); 1876 } catch( NoSuchVariableException e ) { 1877 return null; 1878 } 1879 } 1880 1881 /** 1882 * Throws an exception if a property is not found. 1883 * 1884 * @param props A set of properties to search the key in. 1885 * @param key The key to look for. 1886 * @return The required property 1887 * 1888 * @throws NoRequiredPropertyException If the search key is not in the property set. 1889 * @since 2.0.26 (on TextUtils, moved To WikiEngine on 2.11.0-M1) 1890 */ 1891 public String getRequiredProperty( Properties props, String key ) throws NoRequiredPropertyException { 1892 String value = TextUtil.getStringProperty( props, key, null ); 1893 if( value == null ) { 1894 throw new NoRequiredPropertyException( "Required property not found", key ); 1895 } 1896 return value; 1897 } 1898 1899 /** 1900 * Returns the current PageManager which is responsible for storing 1901 * and managing WikiPages. 1902 * 1903 * @return The current PageManager instance. 1904 */ 1905 public PageManager getPageManager() 1906 { 1907 return m_pageManager; 1908 } 1909 1910 /** 1911 * Returns the CommandResolver for this wiki engine. 1912 * @return the resolver 1913 */ 1914 public CommandResolver getCommandResolver() 1915 { 1916 return m_commandResolver; 1917 } 1918 1919 /** 1920 * Returns the current AttachmentManager, which is responsible for 1921 * storing and managing attachments. 1922 * 1923 * @since 1.9.31. 1924 * @return The current AttachmentManager instance 1925 */ 1926 public AttachmentManager getAttachmentManager() 1927 { 1928 return m_attachmentManager; 1929 } 1930 1931 /** 1932 * Returns the currently used authorization manager. 1933 * 1934 * @return The current AuthorizationManager instance 1935 */ 1936 public AuthorizationManager getAuthorizationManager() 1937 { 1938 return m_authorizationManager; 1939 } 1940 1941 /** 1942 * Returns the currently used authentication manager. 1943 * 1944 * @return The current AuthenticationManager instance. 1945 */ 1946 public AuthenticationManager getAuthenticationManager() 1947 { 1948 return m_authenticationManager; 1949 } 1950 1951 /** 1952 * Returns the manager responsible for the filters. 1953 * @since 2.1.88 1954 * @return The current FilterManager instance 1955 */ 1956 @SuppressWarnings("unchecked") 1957 public < T extends FilterManager > T getFilterManager() 1958 { 1959 return (T)m_filterManager; 1960 } 1961 1962 /** 1963 * Returns the manager responsible for searching the Wiki. 1964 * @since 2.2.21 1965 * @return The current SearchManager instance 1966 */ 1967 public SearchManager getSearchManager() 1968 { 1969 return m_searchManager; 1970 } 1971 1972 /** 1973 * Returns the progress manager we're using 1974 * @return A ProgressManager 1975 * @since 2.6 1976 */ 1977 public ProgressManager getProgressManager() 1978 { 1979 return m_progressManager; 1980 } 1981 1982 /** 1983 * Figure out to which page we are really going to. Considers 1984 * special page names from the jspwiki.properties, and possible aliases. 1985 * This method delgates requests to 1986 * {@link org.apache.wiki.WikiContext#getRedirectURL()}. 1987 * @param context The Wiki Context in which the request is being made. 1988 * @return A complete URL to the new page to redirect to 1989 * @since 2.2 1990 */ 1991 1992 public String getRedirectURL( final WikiContext context ) 1993 { 1994 return context.getRedirectURL(); 1995 } 1996 1997 /** 1998 * Shortcut to create a WikiContext from a supplied HTTP request, 1999 * using a default wiki context. 2000 * @param request the HTTP request 2001 * @param requestContext the default context to use 2002 * @return a new WikiContext object. 2003 * 2004 * @see org.apache.wiki.ui.CommandResolver 2005 * @see org.apache.wiki.ui.Command 2006 * @since 2.1.15. 2007 */ 2008 // FIXME: We need to have a version which takes a fixed page name as well, or check it elsewhere. 2009 public WikiContext createContext( final HttpServletRequest request, final String requestContext ) { 2010 if( !m_isConfigured ) { 2011 throw new InternalWikiException( "WikiEngine has not been properly started. It is likely that the configuration is faulty. Please check all logs for the possible reason." ); 2012 } 2013 2014 // Build the wiki context 2015 final Command command = m_commandResolver.findCommand( request, requestContext ); 2016 return new WikiContext( this, request, command ); 2017 } 2018 2019 /** 2020 * Deletes a page or an attachment completely, including all versions. If the page 2021 * does not exist, does nothing. 2022 * 2023 * @param pageName The name of the page. 2024 * @throws ProviderException If something goes wrong. 2025 */ 2026 public void deletePage( final String pageName ) throws ProviderException { 2027 final WikiPage p = getPage( pageName ); 2028 if( p != null ) { 2029 if( p instanceof Attachment ) { 2030 m_attachmentManager.deleteAttachment( ( Attachment )p ); 2031 } else { 2032 final Collection< String > refTo = m_referenceManager.findRefersTo( pageName ); 2033 // May return null, if the page does not exist or has not been indexed yet. 2034 2035 if( m_attachmentManager.hasAttachments( p ) ) { 2036 final List< Attachment > attachments = m_attachmentManager.listAttachments( p ); 2037 for( final Attachment attachment : attachments ) { 2038 if( refTo != null ) { 2039 refTo.remove( attachment.getName() ); 2040 } 2041 2042 m_attachmentManager.deleteAttachment( attachment ); 2043 } 2044 } 2045 m_pageManager.deletePage( p ); 2046 firePageEvent( WikiPageEvent.PAGE_DELETED, pageName ); 2047 } 2048 } 2049 } 2050 2051 /** 2052 * Deletes a specific version of a page or an attachment. 2053 * 2054 * @param page The page object. 2055 * @throws ProviderException If something goes wrong. 2056 */ 2057 public void deleteVersion( final WikiPage page ) throws ProviderException { 2058 if( page instanceof Attachment ) { 2059 m_attachmentManager.deleteVersion( (Attachment) page ); 2060 } else { 2061 m_pageManager.deleteVersion( page ); 2062 } 2063 } 2064 2065 /** 2066 * Returns the URL of the global RSS file. May be null, if the 2067 * RSS file generation is not operational. 2068 * @since 1.7.10 2069 * @return The global RSS url 2070 */ 2071 public String getGlobalRSSURL() 2072 { 2073 if( m_rssGenerator != null && m_rssGenerator.isEnabled() ) 2074 { 2075 return getBaseURL()+ "/" + m_rssFile; 2076 } 2077 2078 return null; 2079 } 2080 2081 /** 2082 * Returns the root path. The root path is where the WikiEngine is 2083 * located in the file system. 2084 * 2085 * @since 2.2 2086 * @return A path to where the Wiki is installed in the local filesystem. 2087 */ 2088 public String getRootPath() 2089 { 2090 return m_rootPath; 2091 } 2092 2093 /** 2094 * @since 2.2.6 2095 * @return the URL constructor 2096 */ 2097 public URLConstructor getURLConstructor() 2098 { 2099 return m_urlConstructor; 2100 } 2101 2102 /** 2103 * Returns the RSSGenerator. If the property <code>jspwiki.rss.generate</code> 2104 * has not been set to <code>true</code>, this method will return <code>null</code>, 2105 * <em>and callers should check for this value.</em> 2106 * @since 2.1.165 2107 * @return the RSS generator 2108 */ 2109 public RSSGenerator getRSSGenerator() 2110 { 2111 return m_rssGenerator; 2112 } 2113 2114 /** 2115 * Renames, or moves, a wiki page. Can also alter referring wiki 2116 * links to point to the renamed page. 2117 * 2118 * @param context The context during which this rename takes 2119 * place. 2120 * @param renameFrom Name of the source page. 2121 * @param renameTo Name of the destination page. 2122 * @param changeReferrers If true, then changes any referring links 2123 * to point to the renamed page. 2124 * 2125 * @return The name of the page that the source was renamed to. 2126 * 2127 * @throws WikiException In the case of an error, such as the destination 2128 * page already existing. 2129 */ 2130 public String renamePage( WikiContext context, 2131 String renameFrom, 2132 String renameTo, 2133 boolean changeReferrers) 2134 throws WikiException 2135 { 2136 String newPageName = m_pageRenamer.renamePage(context, renameFrom, renameTo, changeReferrers); 2137 firePageRenameEvent(renameFrom, newPageName); 2138 return newPageName; 2139 } 2140 2141 /** 2142 * Returns the PageRenamer employed by this WikiEngine. 2143 * @since 2.5.141 2144 * @return The current PageRenamer instance. 2145 */ 2146 public PageRenamer getPageRenamer() 2147 { 2148 return m_pageRenamer; 2149 } 2150 2151 /** 2152 * Returns the UserManager employed by this WikiEngine. 2153 * @since 2.3 2154 * @return The current UserManager instance. 2155 */ 2156 public UserManager getUserManager() 2157 { 2158 return m_userManager; 2159 } 2160 2161 /** 2162 * Returns the TasksManager employed by this WikiEngine. 2163 * @return The current TasksManager instance. 2164 */ 2165 public TasksManager getTasksManager() 2166 { 2167 return m_tasksManager; 2168 } 2169 2170 /** 2171 * Returns the GroupManager employed by this WikiEngine. 2172 * @since 2.3 2173 * @return The current GroupManager instance 2174 */ 2175 public GroupManager getGroupManager() 2176 { 2177 return m_groupManager; 2178 } 2179 2180 /** 2181 * Returns the current {@link AdminBeanManager}. 2182 * 2183 * @return The current {@link AdminBeanManager}. 2184 * @since 2.6 2185 */ 2186 public AdminBeanManager getAdminBeanManager() { 2187 return m_adminBeanManager; 2188 } 2189 2190 /** 2191 * Returns the AclManager employed by this WikiEngine. 2192 * The AclManager is lazily initialized. 2193 * <p> 2194 * The AclManager implementing class may be set by the 2195 * System property {@link #PROP_ACL_MANAGER_IMPL}. 2196 * </p> 2197 * 2198 * @since 2.3 2199 * @return The current AclManager. 2200 */ 2201 public AclManager getAclManager() 2202 { 2203 if( m_aclManager == null ) 2204 { 2205 try 2206 { 2207 String s = m_properties.getProperty( PROP_ACL_MANAGER_IMPL, ClassUtil.getMappedClass( AclManager.class.getName() ).getName() ); 2208 m_aclManager = ClassUtil.getMappedObject(s); // TODO: I am not sure whether this is the right call 2209 m_aclManager.initialize( this, m_properties ); 2210 } 2211 catch ( ReflectiveOperationException | IllegalArgumentException e ) 2212 { 2213 log.fatal( "unable to instantiate class for AclManager: " + e.getMessage() ); 2214 throw new InternalWikiException( "Cannot instantiate AclManager, please check logs.", e ); 2215 } 2216 } 2217 return m_aclManager; 2218 } 2219 2220 /** 2221 * Returns the DifferenceManager so that texts can be compared. 2222 * @return the difference manager 2223 */ 2224 public DifferenceManager getDifferenceManager() 2225 { 2226 return m_differenceManager; 2227 } 2228 2229 /** 2230 * Returns the current EditorManager instance. 2231 * 2232 * @return The current EditorManager. 2233 */ 2234 public EditorManager getEditorManager() 2235 { 2236 return m_editorManager; 2237 } 2238 2239 /** 2240 * Returns the current i18n manager. 2241 * 2242 * @return The current Intertan... Interante... Internatatializ... Whatever. 2243 */ 2244 public InternationalizationManager getInternationalizationManager() 2245 { 2246 return m_internationalizationManager; 2247 } 2248 2249 /** 2250 * Registers a WikiEventListener with this instance. 2251 * @param listener the event listener 2252 */ 2253 public final synchronized void addWikiEventListener( WikiEventListener listener ) 2254 { 2255 WikiEventManager.addWikiEventListener( this, listener ); 2256 } 2257 2258 /** 2259 * Un-registers a WikiEventListener with this instance. 2260 * @param listener the event listener 2261 */ 2262 public final synchronized void removeWikiEventListener( WikiEventListener listener ) 2263 { 2264 WikiEventManager.removeWikiEventListener( this, listener ); 2265 } 2266 2267 /** 2268 * Fires a WikiEngineEvent to all registered listeners. 2269 * @param type the event type 2270 */ 2271 protected final void fireEvent( int type ) 2272 { 2273 if ( WikiEventManager.isListening(this) ) 2274 { 2275 WikiEventManager.fireEvent(this,new WikiEngineEvent(this,type)); 2276 } 2277 } 2278 2279 /** 2280 * Fires a WikiPageEvent to all registered listeners. 2281 * @param type the event type 2282 */ 2283 protected final void firePageEvent( int type, String pageName ) 2284 { 2285 if ( WikiEventManager.isListening(this) ) 2286 { 2287 WikiEventManager.fireEvent(this,new WikiPageEvent(this,type,pageName)); 2288 } 2289 } 2290 2291 /** 2292 * Fires a WikiPageRenameEvent to all registered listeners. 2293 * @param oldName the former page name 2294 * @param newName the new page name 2295 */ 2296 protected final void firePageRenameEvent(String oldName, String newName ) 2297 { 2298 if ( WikiEventManager.isListening(this) ) 2299 { 2300 WikiEventManager.fireEvent(this,new WikiPageRenameEvent(this,oldName,newName)); 2301 } 2302 } 2303 2304 /** 2305 * Adds an attribute to the engine for the duration of this engine. The 2306 * value is not persisted. 2307 * 2308 * @since 2.4.91 2309 * @param key the attribute name 2310 * @param value the value 2311 */ 2312 public void setAttribute( String key, Object value ) 2313 { 2314 m_attributes.put( key, value ); 2315 } 2316 2317 /** 2318 * Gets an attribute from the engine. 2319 * 2320 * @param key the attribute name 2321 * @return the value 2322 */ 2323 public Object getAttribute( String key ) 2324 { 2325 return m_attributes.get( key ); 2326 } 2327 2328 /** 2329 * Removes an attribute. 2330 * 2331 * @param key The key of the attribute to remove. 2332 * @return The previous attribute, if it existed. 2333 */ 2334 public Object removeAttribute( String key ) 2335 { 2336 return m_attributes.remove( key ); 2337 } 2338 2339 /** 2340 * Returns a WatchDog for current thread. 2341 * 2342 * @return The current thread WatchDog. 2343 * @since 2.4.92 2344 */ 2345 public WatchDog getCurrentWatchDog() 2346 { 2347 return WatchDog.getCurrentWatchDog(this); 2348 } 2349 2350}