J2 Portal Site Component Design

Overview

This document outlines a high to medium level design for the Portal Site component. This component is proposed as the resolution to JIRA issue: JS2-69 Finalizing Portal Navigation using the Profiler.

Generally speaking, the Portal Site component is to use the underlying Page Manager component and Profile Locators generated by the Profiler to create various views of the portal site that can be rendered by the Layout Decorators. The views are composed of profiled reflections of the portal site model comprised of folders, pages, links and other documents, (PSML content). The Portal Site component is responsible for building and managing these views, (i.e. menu, nested menu, separator, and option elements), globally and per request as necessary.

The Portal Site component is also to interpret request URLs processed by the J2 request pipeline and select PSML pages and folders for display in the portal. Generation of the menu based navigation URLs requires the inverse mapping functionality. The Portal Site component utilizes Profile Locators to perform the URL to PSML content mapping.

Existing navigation and URL mapping capabilities currently implemented in the Page Manager components will be deprecated and replaced by the Portal Site component.

High Level Requirements

The following requirements have been gathered from the JS2-69 JIRA issue and other email list sources.

  1. The Portal Site component is responsible for mapping request URLs to underlying PSML content and constructing menu and option views that can be used to navigate to all accessible content for the end user.

  2. In addition to existing page relative menu and option creation, this component must support a simple static menu definition capability that can scale effectively for all portal sites.

  3. Menu definitions should include simple options lists, regular expression wild card options, and depth controlled recursive menu/option PSML reflection.

  4. Menu definitions should be inherited along the profiled folder hierarchy to implement global defaulting and folder specific override capabilities.

  5. Menu definitions must support nested menus and be capable of capturing sufficient information to drive localized pull down menus, (i.e. DHTML javascript site navigation menus).

  6. Menu definitions and this component must provide a method of specifying and generating "bread crumb" navigational aids for menu options.

  7. The Portal Site component must return the current PSML content page and folder within in a logical content hierarchy that can be navigated directly by the Layout Decoration Templates and that reflects the default menu view of the site.

  8. The existing DocumentSet functionality is to be subsumed and deprecated by the menu definition capabilities.

  9. Existing navigations styles must be supported, specifically the page relative "back", "page tabs", "folders", and "additional links" elements.

  10. Existing Portal Site component functionality should be extracted out of Page Manager implementations to promote component reuse and configuration.

  11. The Portal Site component must utilize the standard Page Manager interface to insulate it from various PSML and access security implementations

  12. Folder default page logic and associated history state should be implemented and maintained by the Portal Site component and/or its J2 integration objects.

  13. Global and request specific caching must be implemented to significantly reduce runtime costs of maintaining view elements and must be reset or adjust to changes in the underlying PSML content.

  14. The Portal Site component should compute and cache various content hierarchies and menu views latently; that is only when explicitly requested from the Layout Decorator Templates.

  15. The Portal Site component is instantiated within the J2 Component Architecture, (Spring), and is integrated into the J2 request pipeline and request contexts.

Components and Integration Objects

While the bulk of the functionality of the Portal Site component is to be modularized within the pluggable component itself, two other integration objects are required. The existing Profiler Valve will be extended to attach the Portal Site component to the request pipeline; references within the valve to the Page Manager component will be deprecated by this implementation. The valve will also instantiate and/or update the Portal Site Request Context and Portal Site Session Context objects. These objects are intended to maintain request state for later use by the Layout Decorator Templates and session history for use by the Portal Site component. The request context replaces the existing Profiled Page Context object that is currently referenced within the Layout Decorator Templates using the $site variable.

Figure 1 illustrates the basic relationship between the Portal Site component, its contexts, and other existing J2 components.




Figure 1, Components and Integration Objects

Profiler Valve: instantiates the Portal Site request context with request specific information; retrieves current PSML page and folder selection from the Portal Site request context and injects results into the J2 request context.

Portal Site Request Context: encapsulates request specific state; forwards requests from the Profiler valve and Layout Decorator template to the Portal Site component with the relevant context information; caches request specific results for the Portal Site component.

Portal Site Session Context: maintains default page history for user session used by the Portal Site component to select default pages for folder requests.

Layout Decorator Template: accesses and renders PSML content and view menus/options retrieved from the Portal Site request context; the Portal Site component computes and caches information requested by the templates.

Portal Site Component: accesses the physical PSML content hierarchy using the Page Manager component; selects the current PSML page and folder content, maintaining a logical content hierarchy; generates profiled menu and option views of the PSML site content; caches computed menu views to maximize reuse between requests and sessions.

Page Manager Component: provides PSML access and security implementations; notifies Portal Site component of changes in the PSML content.

Profiler Component: existing component that generates Profile Locators for the current request using defined profiling rules for the user.

General Operation

The high level interaction between these components and the host J2 architecture during a single portal request can be summarized as follows:

  1. The J2 request pipeline invokes the Profiler Valve.

  2. The valve creates a set of Profile Locators associated with the end user and the request being processed using the Profiler component.

  3. If necessary, the valve creates the Portal Site Session Context object in the user's J2 portal session context.

  4. The valve instantiates the Portal Site Request Context object in the J2 request context with the current request, Profile Locators and session context.

  5. The valve then queries the context object for the PSML page and folder to insert into the J2 request context.

  6. To return the appropriate PSML page and folder for the request, the context object queries the Portal Site component with the request and Profile Locators.

  7. The Portal Site component maps the request, Profile Locators, and session default page history into a page and folder from the an appropriate logical content hierarchy derived from the PSML accessed via the Page Manager component.

  8. The logical content hierarchy is cached in the Portal Site component for use in computing menu views or subsequent page and folder queries. Cached content hierarchies are aged using a LRU algorithm or flushed by PSML change notification events received from the Page Manager component.

  9. Resulting logical content page and folder mappings are used by the request context to update the session context default page history while returning the mapping results to the valve.

  10. When the execution of the J2 request pipeline reaches page rendering, the Layout Decorator template is invoked with the context object retrieved from the J2 request context.

  11. The template then accesses the PSML page and folder and/or the menu views via the context object and again forwards these queries to the Portal Site component along with the saved request and Profile Locators.

  12. The Portal Site component selects and computes the requested menu views using the the cached logical content hierarchy and returns the results through the context object to the templates.

  13. Computed menu views are cached in the Portal Site component for subsequent menu view queries. Cached menu views are aged using a LRU algorithm or flushed by change notifications.

PSML Menu Definition Tag Reference

The following PSML Menu Definition is used to define the named menu views to be provided to the Layout Decoration Templates by the Portal Site component. Multiple menu views, (including all standard menu views made available via the existing Profiled Page Context object), are managed by this component. The Menu Definitions are specified in folder.metadata PSML documents. As with other PSML content, the definitions made available for a requested portal page are inherited and/or overridden by name from the current and parent folders. The request Profile Locators are used to aggregate Menu Definitions by name as would be done with the folders they appear in.

Tag Reference
<menu>

Defines a menu or nested menu within a menu.

Attributes:

name - identifies menu name for retrieval from template code and/or menu reference; [required for top level nodes: ignored otherwise]

options - specifies document path for this menu and/or default content elements if not included as explicit children; multiple option paths can be specified as a comma separated list and/or regexp patterns; relative paths interpreted as relative to current folder; special patterns, '~' or '@', can be used to reference the current page; [default: none]

depth - specifies deep inclusion of documents from option roots, (<0 specifies infinite depth); [default: 0]

paths - boolean flag to indicate ordered path options to option value from root; [default: false]

regexp - specifies wild card processing be performed on option value; [default: false]

profile - specified name of Profile Locator to be used when evaluating option value or the default profile value for options and nested menus; specifying '*' forces the acceptance of all Profile Locator names if parent menus select a name; [default: none]

order - comma separated list of regexp patterns matched against list or regexp menu options path values to determine order of matched options; this attribute will be applied as a default options value for any options children, but is not used to reorder multiple options children matches; if not specified, multiple options are included in the order returned by the underlying Page Manager component implementation; option paths not matched by this attribute are appended after ordered matches; [default: none]

skin - optional template defined layout hint and the default skin value for options and nested menus; [default: none]

Children:

title - specifies default locale-independent title for the menu; [1, optional]

short-title - specifies default locale-independent short title for the menu; [1, optional]

metadata - additional meta data specified for the menu, normally used to specify locale-specific titles; [N, optional]

options - specifies content elements for this menu; [N, optional]

menu - nested menu contained in this menu; [N, optional]

separator - specifies text elements for this menu; [N, optional]

include - specifies a menu or menu options to be included in this menu; [N, optional]

exclude - specifies menu options to be excluded from this menu; [N, optional]

<options>

Defines a single or multiple options within a menu.

Attributes:

depth - specifies deep inclusion of documents from option value roots, (<0 specifies infinite depth); [default: 0]

paths - boolean flag to indicate ordered path options to option value from root; [default: false]

regexp - specifies wild card processing be performed on option value; [default: false]

profile - specified name of Profile Locator to be used when evaluating option value; specifying '*' forces the acceptance of all Profile Locator names if parent menus select a name; [default: none]

order - comma separated list of regexp patterns matched against list or regexp options path values to determine order of matched options; if not specified, multiple options are included in the order returned by the underlying Page Manager component implementation; option paths not matched by this attribute are appended after ordered matches; [default: none]

skin - optional template defined layout hint; [default: none]

Children:

[text] - options value; multiple option paths can be specified as a comma separated list and/or regexp patterns; relative paths interpreted as relative to current folder; special patterns, '~' or '@', can be used to reference the current page; [required]

<include>

Includes another menu's options or nests a menu within a menu.

Attributes:

nest - boolean flag that controls whether the specified menu is to be nested; [default: false]

Children:

[text] - identifies included menu name; circular inclusions are suppressed; [required]

<exclude>

Excludes another menu's options or nested menus if they appear above within a menu.

Children:

[text] - identifies excluded menu name; circular exclusions are suppressed; [required]

<separator>

Defines a separator to be included if options or nested menus appear below within a menu.

Attributes:

skin - optional template defined layout hint; [default: none]

Children:

title - specified default locale-independent title for the separator; [1, optional]

metadata - additional meta data specified for the separator text and title, normally used to specify locale-specific text; [N, optional]

text - specified default locale-independent separator text; [1, optional]

[text] - specified default locale-independent separator text; [1, optional]

Examples

A simple menu definition:

<menu name="simple">
   <options>/some-top-page.psml,/custom/some-other-page.psml</options>
</menu>

A top level menu definition that contains all site content down 2 levels deep with DHTML javascript pull down rendering preferred:

<menu name="top-2-levels" options="/" depth="2" skin="dhtml-pull-down"/>

A menu comprised of all top level pages profiled by user role:

<menu name="top-role-pages" regexp="true" options="/*" profile="roles"/>

A menu used to display navigational "bread crumbs" paths:

<menu name="bread-crumbs" options="./" paths="true"/>

A complex nested menu definition:

<menu name="top-custom">
    <title>Top Menu</title>
    <metadata name="title" xml:lang="fr">Haut</metadata>
    <options regexp="true" profile="groups">/group-pages/*</options>
    <menu options="/" profile="page">
        <separator>
            <text>-- Top Pages --</text>
            <title>Top Pages</title>
        </separator>
        <options regexp="true">/*</options>
        <separator>
            <title>Custom Pages</title>
        </separator>
        <options depth="2">/custom-folder/</options>
    </menu>
    <exclude>top-role-pages</exclude>
    <separator>More Top Pages</separator>
    <include nest="true">top-role-pages</include>
</menu>

Standard Menu Definitions

The following standard menu definitions are available for all pages and are intended to replace the existing Profiled Page Context page relative navigation elements. These definitions can be overridden by specifying custom versions of these menus in folder.metadata PSML documents with the same name. Standard and custom menu definitions and views are segregated to facilitate iteration over custom menu definitions.

<menu name="back" options="../"/>

<menu name="breadcrumbs" options="~" paths="true"/>

<menu name="pages" regexp="true" options="*.psml"/>

<menu name="navigations">
  <separator>Folders</separator>
  <options regexp="true">./*/</options>
  <include>page-navigations</include>
  <separator>Additional Links</separator>
  <options regexp="true">/*.link</options>
</menu>

Document Set to Menu Definition Conversion

Document Set PSML support is to be removed as part of the Portal Site component implementation. Here is a PSML tag/attribute mapping that can be used to convert the *.ds documents into the Menu Definition tags in the folder.metadata files:

Document Set

Menu Definition

/document-set

/menu

/document-set/title

/menu/title

/document-set/metadata

/menu/metadata

/document-set/profile-locator

/menu/@profile, /menu//option/@profile

/document-set/document-path

/menu/@option, /menu//option

/document-set/document-path/regexp

/menu/@regexp, /menu//option/@regexp

/document-set/security-constraints

N/A

Table 1, Document Set to Menu Definition Conversion

Component and Integration Interfaces:

The following public interfaces and methods will be developed, modified, or deprecated from component and other object instances as part of the Portal Site component implementation.

PortalSite

PortalSiteSessionContext createSessionContext()
PortalSiteRequestContext createRequestContext()

PortalSiteRequestContext

void setProfileLocators(Map profileLocators)
void setSessionContext(PortalSiteSessionContext context)
Page getPage()
Folder getFolder()
Folder getRootFolder()
Set getStandardMenuNames()
Set getCustomMenuNames()
Menu getMenu(String name)

Map getLocators()
NodeSet getSiblingPages()
Folder getParentFolder()
NodeSet getSiblingFolders()
NodeSet getRootLinks()

PortalSiteSessionContext
DocumentImpl

Node getParent()

FolderImpl

Node getParent()
NodeSet getFolders()
Folder getFolder(String name)
NodeSet getPages()
Page getPage(String name)
NodeSet getLinks()
Link getLink(String name)
PageSecurity getPageSecurity()
NodeSet getAll()

MenuElement

String getElementType()
Menu getParentMenu()
String getTitle()
String getTitle(Locale locale)
String getShortTitle()
String getShortTitle(Locale locale)
GenericMetadata getMetadata()
String getSkin()

Menu

String getName()
String getUrl()
boolean isHidden()
boolean isSelected(PortalSiteRequestContext context)
List getElements()
MenuElement getSelectedElement(PortalSiteRequestContext context)

Option

String getType()
String getUrl()
boolean isHidden()
boolean isSelected(PortalSiteRequestContext context)

Separator

String getText()
String getText(Locale locale)

Profiler

public ProfiledPageContext createProfiledPageContext(Map locators) [delete]

ProfilerValve

void invoke(RequestContext request, ValveContext context)

PageManager

void addPageManagerEventListener(PageManagerEventListener listener)
void removePageManagerEventListener(PageManagerEventListener listener)
void computeProfiledPageContext(ProfiledPageContext pageContext) [delete]
DocumentSet getDocumentSet(String name) [delete]

PageManagerEventListener

void documentChanged(PageManagerEvent event)
void folderChanged(PageManagerEvent event)

PageManagerEvent

Document getDocument()
Folder getFolder()

Folder

NodeSet getDocumentSets() [delete]
DocumentSet getDocumentSet(String name)[delete]

Notes:

Layout Decorator Template Usage

Velocity templates are the preferred implementation used by the J2 decorators. The Portal Site Request Context object API is directly accessible using the $site Velocity context variable. The $site variable currently exports the Profiled Page Context implementation. Even though a compatibility API is being supported by the Portal Site request context, the following conversion for the commonly used Profiled Page Context methods could be applied to existing Layout Decorator templates. Both PSML folder/page and menu/option styles are supported for many of the existing navigation accessors.

ProfiledPageContext method

PortalSiteRequestContext method

$site.page, $site.getPage()

(supported)

$site.folder, $site.getFolder()

(supported)

$site.siblingPages, $site.getSiblingPages()

$site.folder.pages, $site.getFolder().getPages(), $site.getMenu("pages").getElements()

$site.parentFolder, $site.getParentFolder()

$site.folder.parent, $site.getFolder().getParent(), $site.getMenu("back").getElements().get(0)

$site.siblingFolders, $site.getSiblingFolders()

$site.folder.folders, $site.getFolder().getFolders(), $site.getMenu("navigations").getElements()

$site.rootLinks, $site.getRootLinks()

$site.rootFolder.links, $site.getRootFolder().getLinks(), $site.getMenu("navigations").getElements()

$site.getDocumentSet($name)

$site.getMenu($name)

$site.getDocumentSetNodes($name)

$site.getMenu($name).getElements()

$site.documentSetNames, $site.getDocumentSetNames()

$site.customMenuNames, $site.getCustomMenuNames()

Table 2, Profiled Page Context to Portal Site Request Context Conversion

Here is an example of how the Portal Site menu views could be used to generate a "bread crumbs" navigational aid within a Velocity Layout Decorator template.

This example assumes that the following menu definition appears in a folder.metadata within the PSML content:

<menu name="bread-crumbs" option="./" paths="true"/>

The following Velocity fragment would generate a series of HTML anchor tags that would render the menu options into the familiar series of links within the portal layout:

<div class="BreadCrumbs">
#foreach($option in $site.getMenu("bread-crumbs").getElements())
   #if($option.getType() == "folder")
        &nbsp;
        <span>
            <a href="$jetspeed.getAbsoluteUrl($option.url)"
               title="$option.getTitle($preferredLocale)"
               class="BreadCrumbsLink">
                $option.getShortTitle($preferredLocale)
            </a>
        </span>
        &nbsp;
    #end
#end
</div>

Component and Integration Object State

In order for the Portal Site component to perform reasonably, aggressive caching must be implemented to maximize reuse within and across sessions. In addition, request information and intermediate results, (those, for example, related to profiling), are cached in the Portal Site request context object. A catalog of the high level state managed by the components and integration objects is outlined here.

PortalSiteRequestContext
PortalSiteSessionContext
PortalSite
State Definitions:

The logical content search paths are constructed by the Profile Locators which are generated by the Profiler component based upon the end user's profiling rules.

The logical content hierarchies are made up of FolderImpl and DocumentImpl proxies constructed using specific logical content search paths.

The logical folder path is the effective request path to a folder in logical content hierarchies.

The logical default page path is the effective request path to the historical default page in logical content hierarchies.

The menu descriptions are references to PSML objects defined in PSML folder content located via FolderImpl proxies in logical content hierarchies.

The menu views are composed of Menu, Option, and Separator objects instantiated from menu definitions defined from folders in logical content hierarchies.