All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.catalina.ha.session.DeltaRequest Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.catalina.ha.session;

import java.io.ByteArrayOutputStream;
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectOutputStream;
import java.security.Principal;
import java.util.ArrayDeque;
import java.util.Deque;

import org.apache.catalina.SessionListener;
import org.apache.catalina.realm.GenericPrincipal;
import org.apache.juli.logging.Log;
import org.apache.juli.logging.LogFactory;
import org.apache.tomcat.util.res.StringManager;

/**
 * This class is used to track the series of actions that happens when a request is executed. These actions will then
 * translate into invocations of methods on the actual session.
 * 

* This class is NOT thread safe. One DeltaRequest per session. */ public class DeltaRequest implements Externalizable { public static final Log log = LogFactory.getLog(DeltaRequest.class); /** * The string manager for this package. */ protected static final StringManager sm = StringManager.getManager(DeltaRequest.class); public static final int TYPE_ATTRIBUTE = 0; public static final int TYPE_PRINCIPAL = 1; public static final int TYPE_ISNEW = 2; public static final int TYPE_MAXINTERVAL = 3; public static final int TYPE_AUTHTYPE = 4; public static final int TYPE_LISTENER = 5; public static final int TYPE_NOTE = 6; public static final int ACTION_SET = 0; public static final int ACTION_REMOVE = 1; public static final String NAME_PRINCIPAL = "__SET__PRINCIPAL__"; public static final String NAME_MAXINTERVAL = "__SET__MAXINTERVAL__"; public static final String NAME_ISNEW = "__SET__ISNEW__"; public static final String NAME_AUTHTYPE = "__SET__AUTHTYPE__"; public static final String NAME_LISTENER = "__SET__LISTENER__"; private String sessionId; private final Deque actions = new ArrayDeque<>(); private final Deque actionPool = new ArrayDeque<>(); private boolean recordAllActions = false; public DeltaRequest() { } public DeltaRequest(String sessionId, boolean recordAllActions) { this.recordAllActions = recordAllActions; if (sessionId != null) { setSessionId(sessionId); } } public void setAttribute(String name, Object value) { int action = (value == null) ? ACTION_REMOVE : ACTION_SET; addAction(TYPE_ATTRIBUTE, action, name, value); } public void removeAttribute(String name) { addAction(TYPE_ATTRIBUTE, ACTION_REMOVE, name, null); } public void setNote(String name, Object value) { int action = (value == null) ? ACTION_REMOVE : ACTION_SET; addAction(TYPE_NOTE, action, name, value); } public void removeNote(String name) { addAction(TYPE_NOTE, ACTION_REMOVE, name, null); } public void setMaxInactiveInterval(int interval) { addAction(TYPE_MAXINTERVAL, ACTION_SET, NAME_MAXINTERVAL, Integer.valueOf(interval)); } /** * Only support principals from type {@link GenericPrincipal GenericPrincipal} * * @param p Session principal * * @see GenericPrincipal */ public void setPrincipal(Principal p) { int action = (p == null) ? ACTION_REMOVE : ACTION_SET; GenericPrincipal gp = null; if (p != null) { if (p instanceof GenericPrincipal) { gp = (GenericPrincipal) p; if (log.isDebugEnabled()) { log.debug(sm.getString("deltaRequest.showPrincipal", p.getName(), getSessionId())); } } else { log.error(sm.getString("deltaRequest.wrongPrincipalClass", p.getClass().getName())); } } addAction(TYPE_PRINCIPAL, action, NAME_PRINCIPAL, gp); } public void setNew(boolean n) { int action = ACTION_SET; addAction(TYPE_ISNEW, action, NAME_ISNEW, Boolean.valueOf(n)); } public void setAuthType(String authType) { int action = (authType == null) ? ACTION_REMOVE : ACTION_SET; addAction(TYPE_AUTHTYPE, action, NAME_AUTHTYPE, authType); } public void addSessionListener(SessionListener listener) { addAction(TYPE_LISTENER, ACTION_SET, NAME_LISTENER, listener); } public void removeSessionListener(SessionListener listener) { addAction(TYPE_LISTENER, ACTION_REMOVE, NAME_LISTENER, listener); } protected void addAction(int type, int action, String name, Object value) { AttributeInfo info = null; if (this.actionPool.size() > 0) { try { info = actionPool.removeFirst(); } catch (Exception x) { log.error(sm.getString("deltaRequest.removeUnable"), x); info = new AttributeInfo(type, action, name, value); } info.init(type, action, name, value); } else { info = new AttributeInfo(type, action, name, value); } // if we have already done something to this attribute, make sure // we don't send multiple actions across the wire if (!recordAllActions) { try { actions.remove(info); } catch (java.util.NoSuchElementException x) { // do nothing, we wanted to remove it anyway } } // add the action actions.addLast(info); } public void execute(DeltaSession session, boolean notifyListeners) { if (!this.sessionId.equals(session.getId())) { throw new IllegalArgumentException(sm.getString("deltaRequest.ssid.mismatch")); } session.access(); for (AttributeInfo info : actions) { switch (info.getType()) { case TYPE_ATTRIBUTE: if (info.getAction() == ACTION_SET) { if (log.isTraceEnabled()) { log.trace("Session.setAttribute('" + info.getName() + "', '" + info.getValue() + "')"); } session.setAttribute(info.getName(), info.getValue(), notifyListeners, false); } else { if (log.isTraceEnabled()) { log.trace("Session.removeAttribute('" + info.getName() + "')"); } session.removeAttribute(info.getName(), notifyListeners, false); } break; case TYPE_ISNEW: if (log.isTraceEnabled()) { log.trace("Session.setNew('" + info.getValue() + "')"); } session.setNew(((Boolean) info.getValue()).booleanValue(), false); break; case TYPE_MAXINTERVAL: if (log.isTraceEnabled()) { log.trace("Session.setMaxInactiveInterval('" + info.getValue() + "')"); } session.setMaxInactiveInterval(((Integer) info.getValue()).intValue(), false); break; case TYPE_PRINCIPAL: Principal p = null; if (info.getAction() == ACTION_SET) { p = (Principal) info.getValue(); } session.setPrincipal(p, false); break; case TYPE_AUTHTYPE: String authType = null; if (info.getAction() == ACTION_SET) { authType = (String) info.getValue(); } session.setAuthType(authType, false); break; case TYPE_LISTENER: SessionListener listener = (SessionListener) info.getValue(); if (info.getAction() == ACTION_SET) { session.addSessionListener(listener, false); } else { session.removeSessionListener(listener, false); } break; case TYPE_NOTE: if (info.getAction() == ACTION_SET) { if (log.isTraceEnabled()) { log.trace("Session.setNote('" + info.getName() + "', '" + info.getValue() + "')"); } session.setNote(info.getName(), info.getValue(), false); } else { if (log.isTraceEnabled()) { log.trace("Session.removeNote('" + info.getName() + "')"); } session.removeNote(info.getName(), false); } break; default: log.warn(sm.getString("deltaRequest.invalidAttributeInfoType", info)); }// switch } // for session.endAccess(); reset(); } public void reset() { while (actions.size() > 0) { try { AttributeInfo info = actions.removeFirst(); info.recycle(); actionPool.addLast(info); } catch (Exception x) { log.error(sm.getString("deltaRequest.removeUnable"), x); } } actions.clear(); } public String getSessionId() { return sessionId; } public void setSessionId(String sessionId) { this.sessionId = sessionId; if (sessionId == null) { Exception e = new Exception(sm.getString("deltaRequest.ssid.null")); log.error(sm.getString("deltaRequest.ssid.null"), e.fillInStackTrace()); } } public int getSize() { return actions.size(); } public void clear() { actions.clear(); actionPool.clear(); } @Override public void readExternal(java.io.ObjectInput in) throws IOException, ClassNotFoundException { // sessionId - String // recordAll - boolean // size - int // AttributeInfo - in an array reset(); sessionId = in.readUTF(); recordAllActions = in.readBoolean(); int cnt = in.readInt(); for (int i = 0; i < cnt; i++) { AttributeInfo info = null; if (this.actionPool.size() > 0) { try { info = actionPool.removeFirst(); } catch (Exception x) { log.error(sm.getString("deltaRequest.removeUnable"), x); info = new AttributeInfo(); } } else { info = new AttributeInfo(); } info.readExternal(in); actions.addLast(info); } // for } @Override public void writeExternal(java.io.ObjectOutput out) throws IOException { // sessionId - String // recordAll - boolean // size - int // AttributeInfo - in an array out.writeUTF(getSessionId()); out.writeBoolean(recordAllActions); out.writeInt(getSize()); for (AttributeInfo info : actions) { info.writeExternal(out); } } /** * serialize DeltaRequest * * @see DeltaRequest#writeExternal(java.io.ObjectOutput) * * @return serialized delta request * * @throws IOException IO error serializing */ protected byte[] serialize() throws IOException { ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(bos); writeExternal(oos); oos.flush(); oos.close(); return bos.toByteArray(); } private static class AttributeInfo implements Externalizable { private String name = null; private Object value = null; private int action; private int type; AttributeInfo() { this(-1, -1, null, null); } AttributeInfo(int type, int action, String name, Object value) { super(); init(type, action, name, value); } public void init(int type, int action, String name, Object value) { this.name = name; this.value = value; this.action = action; this.type = type; } public int getType() { return type; } public int getAction() { return action; } public Object getValue() { return value; } @Override public int hashCode() { return name.hashCode(); } public String getName() { return name; } public void recycle() { name = null; value = null; type = -1; action = -1; } @Override public boolean equals(Object o) { if (!(o instanceof AttributeInfo)) { return false; } AttributeInfo other = (AttributeInfo) o; return other.getName().equals(this.getName()); } @Override public void readExternal(java.io.ObjectInput in) throws IOException, ClassNotFoundException { // type - int // action - int // name - String // hasvalue - boolean // value - object type = in.readInt(); action = in.readInt(); name = in.readUTF(); boolean hasValue = in.readBoolean(); if (hasValue) { value = in.readObject(); } } @Override public void writeExternal(java.io.ObjectOutput out) throws IOException { // type - int // action - int // name - String // hasvalue - boolean // value - object out.writeInt(getType()); out.writeInt(getAction()); out.writeUTF(getName()); out.writeBoolean(getValue() != null); if (getValue() != null) { out.writeObject(getValue()); } } @Override public String toString() { StringBuilder buf = new StringBuilder("AttributeInfo[type="); buf.append(getType()).append(", action=").append(getAction()); buf.append(", name=").append(getName()).append(", value=").append(getValue()); buf.append(", addr=").append(super.toString()).append(']'); return buf.toString(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy