]> git.argeo.org Git - lgpl/argeo-commons.git/blob - server/runtime/org.argeo.server.jackrabbit/src/main/java/org/argeo/jackrabbit/remote/SimpleSessionProvider.java
Load JCR CND files with OSGi bundle context
[lgpl/argeo-commons.git] / server / runtime / org.argeo.server.jackrabbit / src / main / java / org / argeo / jackrabbit / remote / SimpleSessionProvider.java
1 package org.argeo.jackrabbit.remote;
2
3 import java.io.Serializable;
4 import java.util.ArrayList;
5 import java.util.Collections;
6 import java.util.HashMap;
7 import java.util.Iterator;
8 import java.util.List;
9 import java.util.Map;
10
11 import javax.jcr.LoginException;
12 import javax.jcr.Node;
13 import javax.jcr.Repository;
14 import javax.jcr.RepositoryException;
15 import javax.jcr.Session;
16 import javax.jcr.Value;
17 import javax.servlet.ServletException;
18 import javax.servlet.http.HttpServletRequest;
19
20 import org.apache.commons.logging.Log;
21 import org.apache.commons.logging.LogFactory;
22 import org.apache.jackrabbit.api.JackrabbitSession;
23 import org.apache.jackrabbit.api.security.user.Group;
24 import org.apache.jackrabbit.api.security.user.User;
25 import org.apache.jackrabbit.api.security.user.UserManager;
26 import org.apache.jackrabbit.server.SessionProvider;
27 import org.argeo.ArgeoException;
28 import org.argeo.jcr.ArgeoNames;
29 import org.argeo.jcr.JcrUtils;
30
31 /**
32 * To be injected, typically of scope="session". Implements an open session in
33 * view patter: a new JCR session is created for each request
34 */
35 public class SimpleSessionProvider implements SessionProvider, Serializable {
36 private static final long serialVersionUID = 2270957712453841368L;
37
38 private final static Log log = LogFactory
39 .getLog(SimpleSessionProvider.class);
40
41 private transient Map<String, Session> sessions;
42
43 private Boolean openSessionInView = true;
44
45 private String securityWorkspace = "security";
46
47 public Session getSession(HttpServletRequest request, Repository rep,
48 String workspace) throws LoginException, ServletException,
49 RepositoryException {
50
51 if (openSessionInView) {
52 JackrabbitSession session = (JackrabbitSession) rep
53 .login(workspace);
54 writeRemoteRoles(session);
55 return session;
56 } else {
57 // since sessions is transient it can't be restored from the session
58 if (sessions == null)
59 sessions = Collections
60 .synchronizedMap(new HashMap<String, Session>());
61
62 if (!sessions.containsKey(workspace)) {
63 try {
64 JackrabbitSession session = (JackrabbitSession) rep.login(
65 null, workspace);
66 writeRemoteRoles(session);
67 if (log.isTraceEnabled())
68 log.trace("User " + session.getUserID()
69 + " logged into " + request.getServletPath());
70 sessions.put(workspace, session);
71 return session;
72 } catch (Exception e) {
73 throw new ArgeoException("Cannot open session", e);
74 }
75 } else {
76 Session session = sessions.get(workspace);
77 if (!session.isLive()) {
78 sessions.remove(workspace);
79 session = rep.login(null, workspace);
80 sessions.put(workspace, session);
81 }
82 return session;
83 }
84 }
85 }
86
87 protected void writeRemoteRoles(JackrabbitSession session)
88 throws RepositoryException {
89 if (!session.getWorkspace().getName().equals(securityWorkspace))
90 return;
91
92 // retrieve roles
93 String userId = session.getUserID();
94 UserManager userManager = session.getUserManager();
95 User user = (User) userManager.getAuthorizable(userId);
96 if (user == null) {
97 // anonymous
98 return;
99 }
100 List<String> userGroupIds = new ArrayList<String>();
101 if (user != null)
102 for (Iterator<Group> it = user.memberOf(); it.hasNext();)
103 userGroupIds.add(it.next().getID());
104
105 // write roles if needed
106 Node userProfile = JcrUtils.getUserHome(session).getNode(
107 ArgeoNames.ARGEO_PROFILE);
108 boolean writeRoles = false;
109 if (userProfile.hasProperty(ArgeoNames.ARGEO_REMOTE_ROLES)) {
110 Value[] roles = userProfile.getProperty(
111 ArgeoNames.ARGEO_REMOTE_ROLES).getValues();
112 if (roles.length != userGroupIds.size())
113 writeRoles = true;
114 else
115 for (int i = 0; i < roles.length; i++)
116 if (!roles[i].getString().equals(userGroupIds.get(i)))
117 writeRoles = true;
118 } else
119 writeRoles = true;
120
121 if (writeRoles) {
122 session.getWorkspace().getVersionManager()
123 .checkout(userProfile.getPath());
124 String[] roleIds = userGroupIds.toArray(new String[userGroupIds
125 .size()]);
126 userProfile.setProperty(ArgeoNames.ARGEO_REMOTE_ROLES, roleIds);
127 JcrUtils.updateLastModified(userProfile);
128 session.save();
129 session.getWorkspace().getVersionManager()
130 .checkin(userProfile.getPath());
131 }
132
133 }
134
135 public void releaseSession(Session session) {
136 if (log.isTraceEnabled())
137 log.trace("Releasing JCR session " + session);
138 if (openSessionInView) {
139 if (session.isLive()) {
140 session.logout();
141 if (log.isTraceEnabled())
142 log.trace("Logged out remote JCR session " + session);
143 }
144 }
145 }
146
147 public void init() {
148 }
149
150 public void dispose() {
151 if (sessions != null)
152 for (String workspace : sessions.keySet()) {
153 Session session = sessions.get(workspace);
154 if (session.isLive()) {
155 session.logout();
156 if (log.isDebugEnabled())
157 log.debug("Logged out remote JCR session " + session);
158 }
159 }
160 }
161 }