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 &amp;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, "&nbsp;" );
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 &lt; and &amp; -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 &lt; and &amp; -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 &quot; etc.
1261        //
1262        /*
1263        result = TextUtil.replaceString( result, "&", "&amp;" );
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 &lt; and &amp; -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}