001/* 002 * Copyright 2010 Anyware Services 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016package org.ametys.web.repository.site; 017 018import java.util.ArrayList; 019import java.util.Collection; 020import java.util.HashMap; 021import java.util.HashSet; 022import java.util.Iterator; 023import java.util.Map; 024import java.util.Set; 025 026import javax.jcr.Repository; 027import javax.jcr.RepositoryException; 028import javax.jcr.Session; 029import javax.jcr.observation.Event; 030import javax.jcr.observation.ObservationManager; 031 032import org.apache.avalon.framework.component.Component; 033import org.apache.avalon.framework.context.Context; 034import org.apache.avalon.framework.context.ContextException; 035import org.apache.avalon.framework.context.Contextualizable; 036import org.apache.avalon.framework.logger.AbstractLogEnabled; 037import org.apache.avalon.framework.service.ServiceException; 038import org.apache.avalon.framework.service.ServiceManager; 039import org.apache.avalon.framework.service.Serviceable; 040import org.apache.cocoon.components.ContextHelper; 041import org.apache.cocoon.environment.Request; 042 043import org.ametys.core.right.RightManager; 044import org.ametys.core.user.CurrentUserProvider; 045import org.ametys.core.user.UserIdentity; 046import org.ametys.core.user.population.PopulationContextHelper; 047import org.ametys.plugins.repository.AmetysObject; 048import org.ametys.plugins.repository.AmetysObjectIterable; 049import org.ametys.plugins.repository.AmetysObjectResolver; 050import org.ametys.plugins.repository.AmetysRepositoryException; 051import org.ametys.plugins.repository.ModifiableTraversableAmetysObject; 052import org.ametys.plugins.repository.RepositoryConstants; 053import org.ametys.plugins.repository.RepositoryIntegrityViolationException; 054import org.ametys.plugins.repository.TraversableAmetysObject; 055import org.ametys.plugins.repository.UnknownAmetysObjectException; 056import org.ametys.plugins.repository.provider.AdminSessionProvider; 057import org.ametys.plugins.repository.provider.WorkspaceSelector; 058import org.ametys.web.live.LiveWorkspaceListener; 059import org.ametys.web.repository.page.CopySiteComponent; 060import org.ametys.web.synchronization.SynchronizeComponent; 061 062/** 063 * Helper component for managing sites. 064 */ 065public class SiteManager extends AbstractLogEnabled implements Serviceable, Component, Contextualizable 066{ 067 /** Avalon Role */ 068 public static final String ROLE = SiteManager.class.getName(); 069 070 /** Constant for storing the sites cache in request attribute. */ 071 public static final String SITES_KEY = SiteManager.class.getName(); 072 073 /** sites root's JCR node name */ 074 public static final String ROOT_SITES = RepositoryConstants.NAMESPACE_PREFIX_INTERNAL + ":sites"; 075 076 /** sites root's JCR path */ 077 public static final String ROOT_SITES_PATH = "/" + ROOT_SITES; 078 079 private AmetysObjectResolver _resolver; 080 private Repository _repository; 081 private AdminSessionProvider _adminSessionProvider; 082 private SynchronizeComponent _synchronizeComponent; 083 private CopySiteComponent _copySiteComponent; 084 private WorkspaceSelector _workspaceSelector; 085 private RightManager _rightManager; 086 private CurrentUserProvider _currentUserProvider; 087 private PopulationContextHelper _populationContextHelper; 088 089 private Context _context; 090 091 092 093 @Override 094 public void service(ServiceManager manager) throws ServiceException 095 { 096 _resolver = (AmetysObjectResolver) manager.lookup(AmetysObjectResolver.ROLE); 097 _repository = (Repository) manager.lookup("javax.jcr.Repository"); 098 _adminSessionProvider = (AdminSessionProvider) manager.lookup(AdminSessionProvider.ROLE); 099 _synchronizeComponent = (SynchronizeComponent) manager.lookup(SynchronizeComponent.ROLE); 100 _copySiteComponent = (CopySiteComponent) manager.lookup(CopySiteComponent.ROLE); 101 _workspaceSelector = (WorkspaceSelector) manager.lookup(WorkspaceSelector.ROLE); 102 _rightManager = (RightManager) manager.lookup(RightManager.ROLE); 103 _currentUserProvider = (CurrentUserProvider) manager.lookup(CurrentUserProvider.ROLE); 104 _populationContextHelper = (PopulationContextHelper) manager.lookup(PopulationContextHelper.ROLE); 105 } 106 107 public void contextualize(Context context) throws ContextException 108 { 109 _context = context; 110 } 111 112 private Request _getRequest () 113 { 114 try 115 { 116 return (Request) _context.get(ContextHelper.CONTEXT_REQUEST_OBJECT); 117 } 118 catch (ContextException ce) 119 { 120 getLogger().info("Unable to get the request", ce); 121 return null; 122 } 123 } 124 125 /** 126 * Returns the sites names. 127 * @return the sites names. 128 */ 129 public Collection<String> getRootSiteNames() 130 { 131 TraversableAmetysObject root = _resolver.resolveByPath(ROOT_SITES_PATH); 132 AmetysObjectIterable<AmetysObject> sites = root.getChildren(); 133 134 ArrayList<String> result = new ArrayList<>(); 135 136 for (AmetysObject object : sites) 137 { 138 result.add(object.getName()); 139 } 140 141 return result; 142 } 143 144 /** 145 * Returns the sites names. 146 * @return the sites names. 147 */ 148 public Collection<String> getSiteNames() 149 { 150 ArrayList<String> result = new ArrayList<>(); 151 AmetysObjectIterable<Site> sites = getSites(); 152 153 for (Site site : sites) 154 { 155 result.add(site.getName()); 156 } 157 158 return result; 159 } 160 161 /** 162 * Get the granted site names for the current user 163 * @return The name of sites the current user can access 164 */ 165 public Set<String> getGrantedSites() 166 { 167 return getGrantedSites(_currentUserProvider.getUser()); 168 } 169 170 /** 171 * Get the granted site names for user 172 * @param user The user 173 * @return The name of sites the user can access 174 */ 175 public Set<String> getGrantedSites(UserIdentity user) 176 { 177 Set<String> grantedSiteNames = new HashSet<>(); 178 179 Request request = _getRequest(); 180 181 for (String siteName : getSiteNames()) 182 { 183 if (_populationContextHelper.getUserPopulationsOnContext("/sites/" + siteName, false).contains(user.getPopulationId())) 184 { 185 try 186 { 187 request.setAttribute("siteName", siteName); // Setting temporarily this attribute to check user rights on any object on this site 188 if (!_rightManager.getUserRights(user, "/cms").isEmpty()) 189 { 190 grantedSiteNames.add(siteName); 191 } 192 } 193 finally 194 { 195 request.setAttribute("siteName", null); 196 } 197 } 198 } 199 200 return grantedSiteNames; 201 } 202 203 /** 204 * Determines if the user has granted access to the site 205 * @param user The user 206 * @param siteName The site name 207 * @return <code>true</code> if the user can access to the site 208 */ 209 public boolean isGrantedSite (UserIdentity user, String siteName) 210 { 211 Object oldValue = null; 212 213 Request request = _getRequest(); 214 try 215 { 216 oldValue = request.getAttribute("siteName"); 217 request.setAttribute("siteName", siteName); // Setting temporarily this attribute to check user rights on any object on this site 218 return !_rightManager.getUserRights(user, "/cms").isEmpty(); 219 } 220 finally 221 { 222 request.setAttribute("siteName", oldValue); 223 } 224 } 225 226 /** 227 * Returns the root for sites. 228 * @return the root for sites. 229 */ 230 public ModifiableTraversableAmetysObject getRoot() 231 { 232 return _resolver.resolveByPath(ROOT_SITES_PATH); 233 } 234 235 /** 236 * Returns the root sites. 237 * @return the root sites. 238 */ 239 public AmetysObjectIterable<Site> getRootSites() 240 { 241 TraversableAmetysObject root = _resolver.resolveByPath(ROOT_SITES_PATH); 242 return root.getChildren(); 243 } 244 245 /** 246 * Returns the all sites. 247 * @return the all sites. 248 */ 249 public AmetysObjectIterable<Site> getSites() 250 { 251 String jcrQuery = "//element(*, ametys:site)"; 252 return _resolver.query(jcrQuery); 253 } 254 255 /** 256 * Creates a site with the given name. 257 * @param siteName the site name. 258 * @param parentId the id of the parent site. Can be null. 259 * @return the newly created {@link Site}. 260 * @throws RepositoryIntegrityViolationException if the named site already exists. 261 */ 262 public Site createSite(String siteName, String parentId) throws RepositoryIntegrityViolationException 263 { 264 ModifiableTraversableAmetysObject root = null; 265 if (parentId != null) 266 { 267 Site site = _resolver.resolveById(parentId); 268 if (!site.hasChild(ROOT_SITES)) 269 { 270 site.createChild(ROOT_SITES, "ametys:sites"); 271 } 272 root = site.getChild(ROOT_SITES); 273 } 274 else 275 { 276 root = _resolver.resolveByPath(ROOT_SITES_PATH); 277 } 278 279 Site site = (Site) root.createChild(siteName, "ametys:site"); 280 try 281 { 282 String sitePath = site.getNode().getPath(); 283 284 // Create the resource explorer root. 285 site.getNode().addNode(RepositoryConstants.NAMESPACE_PREFIX_INTERNAL + ":resources", "ametys:resources-collection"); 286 287 Session session = _adminSessionProvider.getAdminSession(); 288 ObservationManager manager = session.getWorkspace().getObservationManager(); 289 290 manager.addEventListener(new LiveWorkspaceListener(_repository, _synchronizeComponent, getLogger()), 291 Event.NODE_ADDED 292 | Event.NODE_REMOVED 293 | Event.NODE_MOVED 294 | Event.PROPERTY_ADDED 295 | Event.PROPERTY_CHANGED 296 | Event.PROPERTY_REMOVED, 297 sitePath + "/ametys-internal:plugins", true, null, null, false); 298 299 manager.addEventListener(new LiveWorkspaceListener(_repository, _synchronizeComponent, getLogger()), 300 Event.NODE_ADDED 301 | Event.NODE_REMOVED 302 | Event.NODE_MOVED 303 | Event.PROPERTY_ADDED 304 | Event.PROPERTY_CHANGED 305 | Event.PROPERTY_REMOVED, 306 sitePath + "/ametys-internal:resources", true, null, null, false); 307 } 308 catch (RepositoryException ex) 309 { 310 throw new AmetysRepositoryException(ex); 311 } 312 313 return site; 314 } 315 316 /** 317 * Creates a site with the given name. 318 * @param siteName the site name. 319 * @return the newly created {@link Site}. 320 * @throws RepositoryIntegrityViolationException if the named site already exists. 321 */ 322 public Site createSite(String siteName) throws RepositoryIntegrityViolationException 323 { 324 return createSite (siteName, null); 325 } 326 327 /** 328 * Creates a site with the given name, from another site to copy 329 * @param site the site to be copied 330 * @param siteName the site name 331 * @return the newly created {@link Site}. 332 * @throws RepositoryIntegrityViolationException if the named site already exists. 333 */ 334 public Site copySite(Site site, String siteName) throws RepositoryIntegrityViolationException 335 { 336 return copySite(site, null, siteName); 337 } 338 339 /** 340 * Creates a site with the given name, from another site to copy 341 * @param site the site to be copied 342 * @param parentId the id of the parent site. Can be null. 343 * @param siteName the site name 344 * @return the newly created {@link Site}. 345 * @throws RepositoryIntegrityViolationException if the named site already exists. 346 */ 347 public Site copySite(Site site, String parentId, String siteName) throws RepositoryIntegrityViolationException 348 { 349 ModifiableTraversableAmetysObject root = null; 350 if (parentId != null) 351 { 352 root = _resolver.resolveById(parentId); 353 } 354 else 355 { 356 root = _resolver.resolveByPath(ROOT_SITES_PATH); 357 } 358 359 Site cSite = site.copyTo(root, siteName); 360 361 // Clear url and title 362 cSite.getMetadataHolder().removeMetadata("url"); 363 cSite.getMetadataHolder().removeMetadata("title"); 364 365 // Change reference to ametys object, re-init contents workflow, update site name, ... 366 _copySiteComponent.updateSiteAfterCopy(site, cSite); 367 368 return cSite; 369 370 } 371 372 373 /** 374 * Returns true if the given site exists. 375 * @param siteName the site name. 376 * @return true if the given site exists. 377 */ 378 public boolean hasSite(String siteName) 379 { 380 if (siteName == null) 381 { 382 return false; 383 } 384 385 Request request = _getRequest(); 386 if (request == null) 387 { 388 // There is no request to store cache 389 return _hasSite(siteName); 390 } 391 392 @SuppressWarnings("unchecked") 393 Map<String, Site> sitesCache = (Map<String, Site>) request.getAttribute(SITES_KEY); 394 if (sitesCache == null) 395 { 396 sitesCache = new HashMap<>(); 397 request.setAttribute(SITES_KEY, sitesCache); 398 } 399 400 // The site key in the cache is of the form "site/workspace". 401 String siteKey = siteName + "/" + _workspaceSelector.getWorkspace(); 402 403 Site site = sitesCache.get(siteKey); 404 if (site != null) 405 { 406 // Site is in cache 407 return true; 408 } 409 410 site = _getSite(siteName); 411 if (site != null) 412 { 413 // Store site in cache 414 sitesCache.put(siteKey, site); 415 return true; 416 } 417 418 return false; 419 } 420 421 private boolean _hasSite (String siteName) 422 { 423 String jcrQuery = "//element(" + siteName + ", ametys:site)"; 424 AmetysObjectIterable<Site> sites = _resolver.query(jcrQuery); 425 426 return sites.iterator().hasNext(); 427 } 428 429 /** 430 * Returns the named {@link Site}. 431 * @param siteName the site name. 432 * @return the named {@link Site}. 433 * @throws UnknownAmetysObjectException if the named site does not exist. 434 */ 435 public Site getSite(String siteName) throws UnknownAmetysObjectException 436 { 437 Request request = _getRequest(); 438 if (request == null) 439 { 440 // There is no request to store cache 441 return _getSite(siteName); 442 } 443 444 @SuppressWarnings("unchecked") 445 Map<String, Site> sitesCache = (Map<String, Site>) request.getAttribute(SITES_KEY); 446 if (sitesCache == null) 447 { 448 sitesCache = new HashMap<>(); 449 request.setAttribute(SITES_KEY, sitesCache); 450 } 451 452 // The site key in the cache is of the form "site/workspace". 453 String siteKey = siteName + "/" + _workspaceSelector.getWorkspace(); 454 455 Site site = sitesCache.get(siteKey); 456 if (site == null) 457 { 458 site = _getSite(siteName); 459 if (site != null) 460 { 461 // Store site in cache 462 sitesCache.put(siteKey, site); 463 } 464 } 465 466 return site; 467 } 468 469 /** 470 * Clear the site cache 471 */ 472 public void clearCache () 473 { 474 Request request = _getRequest(); 475 request.setAttribute(SITES_KEY, null); 476 } 477 478 private Site _getSite (String siteName) throws UnknownAmetysObjectException 479 { 480 String jcrQuery = "//element(" + siteName + ", ametys:site)"; 481 AmetysObjectIterable<Site> sites = _resolver.query(jcrQuery); 482 Iterator<Site> it = sites.iterator(); 483 484 if (it.hasNext()) 485 { 486 return it.next(); 487 } 488 489 return null; 490 } 491}