Users of eZ Publish sometimes wonder if it is possible to make use of a single installation of eZ Publish for multiple websites. The answer is yes—through the power of the “siteaccess”. In fact, an understanding of siteaccesses gives one the ability to do much more than just host multiple sites in one installation. With this technique you can:
This article written for intermediate-level eZ Publish developers, who already know how to install eZ Publish, create a design, program in the template language, and make use of the settings cascade. These skills can be gained by taking an eZ Publish Developer Basics training course, reading eZ Publish Basics or other eZ Publish books, or learning from a solid eZ Publish developer. As well, the community is a vast source of knowledge and exchange. It can be found here : http://share.ez.no.
The tutorial example in this article is based on this installation:
Let's discuss a variety of reasons one might want several siteaccesses, explore what one is, and then see how it's done. Here are some typical scenarios for using different siteaccesses in addition to the admin and public siteaccess:
Case | Siteaccesses | Content-Sharing Capabilities | Details |
typical (e.g. a single web shop, social networking site, or corporate site) |
site_public site_admin |
n/a (only one real website) | both share the same database and var directory but use different designs and extensions |
multilingual (e.g. a corporate website with three translations plus an administrative interface) |
nor slk chi admin |
all sites can access the same content provided there's an available translation | same database and var directory and same design (except admin), but different language settings in each |
several completely different websites (e.g. a company which sells websites with similar features but different design and content to companies in an industry) |
site_1_public site_1_admin site_2_public site_2_admin ... |
Content (and user accounts) are kept separate so there is no danger of one customer accessing another customer's data. | different databases, var directories, and designs |
one site which gets customized for different populations (for example, a magazine publisher with different versions for various cities) |
city_1_public city_1_admin city_2_public city_2_admin ... |
Content is kept separate for each edition. | design may be shared but databases and var directories may be distinct |
Several sites revolving around a brand/company/idea cross-sharing part of their content, functionalities, and design with each other. |
brand_site_1_public brand_site_1_admin brand_site_2_public brand_site_2_admin |
Content is partly kept separate for common editors: one subtree per site, but can be shared easily for enabled editors. | Design may be shared,extensions too, database and var directories as well. One content subtree is one site. |
Feel free to add your own creative uses of siteaccesses in the related forum here.
Each siteaccess isn't exactly a different website, though it can certainly create that effect. A siteaccess has two main elements:
Let's say you already have an eZ Publish installation. You're used to addressing the homepage as http://www.example.com/index.php/eng. The “eng” is your siteaccess. Perhaps you've rid yourself of the “eng” in your url by making it your default. Or instead of URI-based access, you've switched to host-based access, and the “www” specifies your siteaccess. Either way, the siteaccess is specified in your URL.
You already have a second siteaccess. When you log in to the admin interface, you use http://www.example.com/index.php/admin, or perhaps http://admin.example.com/index.php. In both cases, the siteaccess requested is “admin”.
At first glance you might have thought the admin interface is really a separate piece of software. But it's simply another siteaccess. In other words, it's just a way of accessing your installation by specifying a different set of configuration files. Remember—in eZ Publish, configuration files let you control everything you want, plus scores of things you had never even thought about. So one siteaccess can produce very different results from another.
For example, you could have different designs used, different language settings, or different extensions with different modules, workflows, datatypes, etc. Moreover, you can even specify that you want a different database used, so that none of the content or even class definitions would be shared.
This is an example of how to change your existing ezwebin eZ Publish installation from one public and one admin siteaccesses (named “public” and “admin”) to two public and one admin siteaccesses. Each public site will have its own landing page node. We will specify those node ID's in settings, but first we must create the landing page (home page) for each site in the content node tree. The current “Home” object (of the 'Frontpage' content class) will be reused as one of the two home pages, but we will need to create the second one.
For simplicity of demonstration, we will remove all content than the “Websites” folder and the two home pages created above.
Now we may edit our settings. Node two, previously the landing page node for the one public siteaccess, has become the parent (container) for the two new landing page nodes (see how-to part I). Let's say existing siteaccess 'public' happens to have as its landing page node 175, and siteaccess 'new_site' has landing page node is 176.
1. Edit settings/siteaccess/public/content.ini.append.php :
[NodeSettings] RootNode=175
2. Edit settings/siteaccess/public/site.ini.append.php :
[SiteSettings] SiteName=Public site one SiteURL=localhost/ez-4.3-webin/index.php/public SiteURL=example.com/index.php/public # (or public_site.example.com) ? IndexPage=/content/view/full/175 DefaultPage=/content/view/full/175 RootNodeDepth=2 # customize the metadata by site if you like : MetaDataArray[author]=John Doe MetaDataArray[copyright]=John Doe co.ltd MetaDataArray[description]=This is my public website number one, out of two on the same eZ instance MetaDataArray[keywords]=Public website one, multi-siteaccesses with eZ Publish
This is the beginning of leveraging eZ Publish's Search Engine Optimization (SEO) features. Feel free to do more. Extensions which may be of help include Meta Data Datatyp, Google Sitemaps, Google News Sitemap, and feZ Meta Data.
[SiteAccessSettings] PathPrefix=Public-site-one # (name of node 175 if you don't want its name appearing in the URL) # Comment out the following array reset line, will ease the addition # of new siteaccesses in our tutorial. # RelatedSiteAccessList[]
3. Copy settings/siteaccess/public to settings/siteaccess/new_site
4. Repeat steps 1 and 2 for new_site (replacing 175 with 176, and using the proper url)
5. Edit settings/override/site.ini.append.php :
Make sure none of the above settings is in override/site.ini.append.php. If any is, comment it out and make the appropriate addition to settings/siteaccess/admin/site.ini.append.php. Also in the override site.ini.append.php, add :
[SiteSettings] SiteList[]=new_site [SiteAccessSettings] AvailableSiteAccessList[]=new_site RelatedSiteAccessList[]=new_site PathPrefixExclude[] PathPrefixExclude[]=Media PathPrefixExclude[]=Users
Note that in a site using host-based access method, we'd add something like:
HostMatchMapItems[]=new_site.example.com;new_site
If you're using Website Interface (ezwebin) 1.6 and have the top menu bar, edit menu/flat_top.tpl so it knows where to find the list of pages below the landing page node:
{def $root_node=fetch( 'content', 'node', hash( 'node_id', $indexpage ) )
to
{def $root_node=fetch( 'content', 'node', hash( 'node_id', ezini( 'NodeSettings', 'RootNode', 'content.ini' ) ) )
The change above leads to the following visual change :
Edit permissions for the anonymous user to allow login to the new siteaccess.
Now it is possible to add content under each home page. It will then appear only on the correct website.
If you'd like the search function to only search the subtree below each siteaccess's own landing page node, add the following to pagelayout.tpl's search form:
<input type="hidden" name="SubTreeArray[]" value="{$indexpage}" />
Do similarly for templates/content/advancedsearch.tpl and templates/content/search.tpl, though you'll want to use something more like “value="{ezini( 'NodeSettings', 'RootNode', 'content.ini' )}” since $indexpage isn't available there.
Clear caches for changes to take effect if you have not turned your eZ Publish instance into development mode.
You can also then have each siteaccess produce a sitemap of only its own site. Edit the template showing the series of links at the top-right corner of the page to make sure the generated links take the new content root into account. It is called page_header_links.tpl in the 'ezwebin' design.
The generated sitemap URLs will say something like :
http://www.example.com/index.php/public/content/view/sitemap/175 (or 176 for the other siteaccess)
If you have a Google sitemap or Google news sitemap generator, make the appropriate adjustment there as well.
If you are making use of the “site settings” feature (which may be accessed on the public site by a link near “registration” and “login”, or in the administration interface's Design/Look and Feel), it will work well for websites which are multiple translations of the same data. This feature is useful for demo's and small sites. However, it will not work by default for different websites with different data (whether using multiple databases or different subtrees of the same content tree). In this case, it is generally best to modify the templates which use this data to draw it from a content nodes you specify for each site separately.
This how-to was certainly not exhaustive. Your own site may have other features which get broken by trying to switch from one root node to two landing page nodes beneath a folder. The example code above should give good hints as to how to fix your custom template code.
Your next step might be to create a custom design for each site by modifying the DesignSettings block in settings/siteaccess/<siteaccess-name>/site.ini.append.php. Note that if you keep your design settings in an extension, you'll have to list that extension in settings/override/site.ini.append.php; it's not enough to simply list it as an ActiveAccessExtension in settings/siteaccess/<siteaccess-name>/site.ini.append.php. The ActiveAccessExtension setting is useful because it exposes only the necessary URL's and API's. It allows each site to use a different design extension. Alternately, siteaccesses could share one common design extension, and each have their own specific design extension cascading over the common design extension.
Putting this all together, if we created a “publicdesign” design in a “publicdesign” extension, and a “newsitedesign” in a “newsitedesign” extnesion, and also an “ourshareddesign” in an “ourshareddesign” extension, we could see some example ini settings. For the “public” siteaccess this would look like:
[ExtensionSettings] ActiveAccessExtensions[]=publicdesign ActiveAccessExtensions[]=ourshareddesign [DesignSettings] SiteDesign=publicdesign AdditionalDesignList[]=ourshareddesign AdditionalDesignList[]=ezwebin AdditionalDesignList[]=base AdditionalDesignList[]=standard
On “new_site” we would do this:
[ExtensionSettings] ActiveAccessExtensions[]=newsitedesign ActiveAccessExtensions[]=ourshareddesign [DesignSettings] SiteDesign=newsitedesign AdditionalDesignList[]=ourshareddesign AdditionalDesignList[]=ezwebin AdditionalDesignList[]=base AdditionalDesignList[]=standard
In this example we've used the same database and var directory for both public sites. Though the content is in separate subtrees, the content classes they use are shared. Be careful not to change attributes in a class to suit one site if they're needed by the other site. If this might become a problem, you could use a separate database and var directory—or more simply, each site could use different content class groups.
Copying an object or subtree is of course a simple way of getting content into a second subtree (website) when the database is shared by two sites, but then changes to one copy will not appear on the other. In the same scenario, you can easily share content by using secondary/additional locations. That's an easy way of sharing content, but it must be done for each individual object (the children of the shared object do not automatically get shared as well). A workflow (“after publishing”) could be used to publish all articles in one subtree in another subtree (on the second site) as well. Another way to pull content from one database to another would be using an RSS feed.
It is more secure to use separate databases and var directories for separate sites. It may be best to always use separate databases and var directories unless the siteaccesses need access to some of the same content or users in an easy way. If you do have a database and var directory for each public siteaccess, each public siteaccess will need a corresponding admin siteaccess.
A Single Sign On (SSO) capability is sometimes required, letting a visitor log in with one single set of access credentials on a series of websites. This can be easily achieved when the content is shared between siteaccesses (one subtree per site), for the users (stored as other content objects) are natively shared. When the content base is not shared, dedicated SSO solutions, such as LDAP or Active Directory, must be used. eZ Publish natively supports LDAP as an SSO system, and its architecture can make use of any SSO-plugin. The latter take the form of an extension, and can interface eZ Publish with basically any SSO system. More on SSO extensions here : http://share.ez.no/articles/ez-publish/using-a-sso-in-ez-publish.
Concerning access control, user roles can be applied with subtree limitations, and we now know each website can be a subtree. In our example, care would need to be given to ensure the node 2 is not tampered with, and that the name of the landing page nodes is not changed without an appropriate configuration file change. This can be assured by changing user roles and policies in the administration interface. Having roles applied with subtree limitations eases access control administration. This generic benefit makes a whole lot of sense in our example : the few user profiles are identified, across the two public sites we built, then materialized as roles. Typically, a logged-in user on both site 1 and site 2 can read content from the standard section, the media library, and create & edit blog posts. However, they should not be able to read or edit content from the other website (ie: form the other subtree). To satisfy this :
Imagine that your new, award-winning website (siteaccess “new_site”) has been well so well received that your boss now wants to have it available in Chinese as well. We simply add another siteaccess (“chinese_new_site”) which will use the same data, but with different language settings.
1. Copy the 'new_site' directory under settings/siteaccess, and rename it to 'chinese_new_site'.
2. Edit settings/siteaccess/chinese_new_site/site.ini.append.php to show only Chinese:
[RegionalSettings] TextTranslation=enabled Locale=chi-CN ContentObjectLocale=chi-CN SiteLanguageList[]=chi-CN ShowUntranslatedObjects=disabled
3. Alter access-related settings :
[SiteSettings] SiteName=Public site two in CHINESE SiteURL=localhost/ez-4.3-webin/index.php/chinese_new_site SiteURL=example.com/index.php/ chinese_new_site # (or chinese_new_site.example.com) ? IndexPage=/content/view/full/176 DefaultPage=/content/view/full/176 RootNodeDepth=2 MetaDataArray[author]=John Doe MetaDataArray[copyright]=John Doe co.ltd MetaDataArray[description]=This is my public website number two in CHINESE, out of two on the same eZ instance MetaDataArray[keywords]=Public website two in CHINESE, multi-siteaccesses with eZ Publish
4. Edit settings/override/site.ini.append.php, adding
[SiteSettings] SiteList[]=chinese_new_site [SiteAccessSettings] AvailableSiteAccessList[]=chinese_new_site RelatedSiteAccessList[]=chinese_new_site HostMatchMapItems[]=chinese.example.com;chinese_new_site
5. Add chinese as a language for editors : navigate to Setup > Languages from the administration interface.
6. Allow anonymous login to this new siteaccess (make a ref to same procedure earlier above)
7. Clear the caches
Here is the final result after having added some content :
eZ Publish is already an incredibly configurable framework / content management platform. Siteaccesses allow one to offer multiple sets of configurations in one installation, using the same or different data. This allows one site to be offered in different languages, or multiple sites to be run in one installation, or different looks to be offered with the same data. Adding an additional siteaccess requires careful planning; the basic steps for an example site were provided here.
Here are some links to other reading you might find helpful.
This article is available in PDF for offline reading :
Greg McAvoy Jensen - Lots of Websites One eZ Publish Installation Adding Siteaccesses in eZ Publish - PDF
Greg is the executive director of Granite Horizon, one of the largest eZ Publish development firms in the United States of America. A certified eZ Publish developer, he has served clients and trained developers on four continents since he started working with eZ Publish in 2003. His wife Heather and he are proud parents of three wonderful children.
Timing: | Jan 18 2025 02:10:45 |
Script start | |
Timing: | Jan 18 2025 02:10:45 |
Module start 'layout' | |
Timing: | Jan 18 2025 02:10:45 |
Module start 'content' | |
Warning: XML output handler: link | Jan 18 2025 02:10:45 |
Current user does not have read access to the object of node #95050 | |
Warning: XML output handler: link | Jan 18 2025 02:10:45 |
Current user does not have read access to the object of node #95053 | |
Warning: XML output handler: link | Jan 18 2025 02:10:45 |
Current user does not have read access to the object of node #95054 | |
Warning: XML output handler: link | Jan 18 2025 02:10:45 |
Current user does not have read access to the object of node #95056 | |
Warning: XML output handler: link | Jan 18 2025 02:10:45 |
Current user does not have read access to the object of node #95057 | |
Timing: | Jan 18 2025 02:10:45 |
Module end 'content' | |
Timing: | Jan 18 2025 02:10:45 |
Script end |
Total runtime | 0.3277 sec |
Peak memory usage | 4,096.0000 KB |
Database Queries | 91 |
Checkpoint | Start (sec) | Duration (sec) | Memory at start (KB) | Memory used (KB) |
---|---|---|---|---|
Script start | 0.0000 | 0.0056 | 592.0391 | 152.6875 |
Module start 'layout' | 0.0056 | 0.0027 | 744.7266 | 39.5313 |
Module start 'content' | 0.0082 | 0.3181 | 784.2578 | 1,210.5859 |
Module end 'content' | 0.3263 | 0.0013 | 1,994.8438 | 44.5859 |
Script end | 0.3277 | 2,039.4297 |
Accumulator | Duration (sec) | Duration (%) | Count | Average (sec) |
---|---|---|---|---|
Ini load | ||||
Load cache | 0.0030 | 0.9241 | 16 | 0.0002 |
Check MTime | 0.0012 | 0.3781 | 16 | 0.0001 |
Mysql Total | ||||
Database connection | 0.0007 | 0.2011 | 1 | 0.0007 |
Mysqli_queries | 0.0754 | 23.0160 | 91 | 0.0008 |
Looping result | 0.0007 | 0.2150 | 89 | 0.0000 |
Template Total | 0.3028 | 92.4 | 2 | 0.1514 |
Template load | 0.0016 | 0.5004 | 2 | 0.0008 |
Template processing | 0.3011 | 91.8794 | 2 | 0.1506 |
Template load and register function | 0.0001 | 0.0274 | 1 | 0.0001 |
states | ||||
state_id_array | 0.0066 | 2.0037 | 14 | 0.0005 |
state_identifier_array | 0.0055 | 1.6764 | 15 | 0.0004 |
Override | ||||
Cache load | 0.0049 | 1.4964 | 379 | 0.0000 |
Sytem overhead | ||||
Fetch class attribute name | 0.0022 | 0.6862 | 24 | 0.0001 |
Fetch class attribute can translate value | 0.0001 | 0.0391 | 12 | 0.0000 |
class_abstraction | ||||
Instantiating content class attribute | 0.0000 | 0.0103 | 24 | 0.0000 |
XML | ||||
Image XML parsing | 0.0179 | 5.4650 | 12 | 0.0015 |
General | ||||
dbfile | 0.0147 | 4.5005 | 62 | 0.0002 |
String conversion | 0.0000 | 0.0027 | 4 | 0.0000 |
Note: percentages do not add up to 100% because some accumulators overlap |
Usage | Requested template | Template | Template loaded | Edit | Override |
---|---|---|---|---|---|
1 | node/view/full.tpl | full/article.tpl | extension/sevenx/design/simple/override/templates/full/article.tpl | ||
12 | content/datatype/view/ezxmltext.tpl | <No override> | extension/community_design/design/suncana/templates/content/datatype/view/ezxmltext.tpl | ||
83 | content/datatype/view/ezxmltags/paragraph.tpl | <No override> | extension/ezwebin/design/ezwebin/templates/content/datatype/view/ezxmltags/paragraph.tpl | ||
29 | content/datatype/view/ezxmltags/separator.tpl | <No override> | extension/community_design/design/suncana/templates/content/datatype/view/ezxmltags/separator.tpl | ||
23 | content/datatype/view/ezxmltags/header.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/header.tpl | ||
17 | content/datatype/view/ezxmltags/li.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/li.tpl | ||
7 | content/datatype/view/ezxmltags/ul.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/ul.tpl | ||
23 | content/datatype/view/ezxmltags/link.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/link.tpl | ||
12 | content/datatype/view/ezxmltags/newpage.tpl | <No override> | extension/community/design/standard/templates/content/datatype/view/ezxmltags/newpage.tpl | ||
16 | content/datatype/view/ezxmltags/strong.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/strong.tpl | ||
19 | content/datatype/view/ezxmltags/td.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/td.tpl | ||
6 | content/datatype/view/ezxmltags/tr.tpl | <No override> | extension/community/design/community/templates/content/datatype/view/ezxmltags/tr.tpl | ||
10 | content/datatype/view/ezxmltags/line.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/line.tpl | ||
7 | content/datatype/view/ezxmltags/emphasize.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/emphasize.tpl | ||
1 | content/datatype/view/ezxmltags/table.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/table.tpl | ||
12 | content/datatype/view/ezxmltags/embed.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/embed.tpl | ||
12 | content/view/embed.tpl | embed/image.tpl | extension/sevenx/design/simple/override/templates/embed/image.tpl | ||
12 | content/datatype/view/ezimage.tpl | <No override> | extension/sevenx/design/simple/templates/content/datatype/view/ezimage.tpl | ||
13 | content/datatype/view/ezxmltags/literal.tpl | <No override> | extension/community/design/standard/templates/content/datatype/view/ezxmltags/literal.tpl | ||
1 | content/datatype/view/ezxmltags/embed-inline.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/embed-inline.tpl | ||
1 | content/view/embed-inline.tpl | <No override> | design/standard/templates/content/view/embed-inline.tpl | ||
1 | print_pagelayout.tpl | <No override> | extension/community/design/community/templates/print_pagelayout.tpl | ||
Number of times templates used: 318 Number of unique templates used: 22 |
Time used to render debug report: 0.0002 secs