Я развертываю приложение в кластере WebLogic 10.3.2 с двумя узлами и балансировщиком нагрузки перед кластером.
Я установил
<core:init distributable="true" debug="true" />
Мои классы Session и Conversation реализуют Serializable.
Я начинаю использовать приложение, обслуживаемое первым узлом. Консоль показывает, что репликация сеанса работает.
<Jun 17, 2010 11:43:50 AM EEST> <Info> <Cluster> <BEA-000128> <Updating 5903057688359791237S:xxx.yyy.gr:[7002,7002,-1,-1,-1,-1,-1]:xxx.yyy.gr:7002,xxx.yyy.gr:7002:prs_domain:PRS_Server_2 in the cluster.>
<Jun 17, 2010 11:43:50 AM EEST> <Info> <Cluster> <BEA-000128> <Updating 5903057688359791237S:xxx.yyy.gr:[7002,7002,-1,-1,-1,-1,-1]:xxx.yyy.gr:7002,xxx.yyy.gr:7002:prs_domain:PRS_Server_2 in the cluster.>
Когда я выключаю первый узел из консоли администрирования, я получаю это на другом узле:
<Jun 17, 2010 11:23:46 AM EEST> <Error> <Kernel> <BEA-000802> <ExecuteRequest failed
java.lang.NullPointerException.
java.lang.NullPointerException
at org.jboss.seam.intercept.JavaBeanInterceptor.callPostActivate(JavaBeanInterceptor.java:165)
at org.jboss.seam.intercept.JavaBeanInterceptor.invoke(JavaBeanInterceptor.java:73)
at com.myproj.beans.SortingFilteringBean_$$_javassist_seam_2.sessionDidActivate(SortingFilteringBean_$$_javassist_seam_2.java)
at weblogic.servlet.internal.session.SessionData.notifyActivated(SessionData.java:2258)
at weblogic.servlet.internal.session.SessionData.notifyActivated(SessionData.java:2222)
at weblogic.servlet.internal.session.ReplicatedSessionData.becomePrimary(ReplicatedSessionData.java:231)
at weblogic.cluster.replication.WrappedRO.changeStatus(WrappedRO.java:142)
at weblogic.cluster.replication.WrappedRO.ensureStatus(WrappedRO.java:129)
at weblogic.cluster.replication.LocalSecondarySelector$ChangeSecondaryInfo.run(LocalSecondarySelector.java:542)
at weblogic.work.SelfTuningWorkManagerImpl$WorkAdapterImpl.run(SelfTuningWorkManagerImpl.java:516)
at weblogic.work.ExecuteThread.execute(ExecuteThread.java:201)
at weblogic.work.ExecuteThread.run(ExecuteThread.java:173)
>
Что я делаю не так?
Это SortingFilteringBean:
import java.util.HashMap;
import java.util.LinkedHashMap;
import org.jboss.seam.ScopeType;
import org.jboss.seam.annotations.Name;
import org.jboss.seam.annotations.Scope;
import com.myproj.model.crud.Filtering;
import com.myproj.model.crud.Sorting;
import com.myproj.model.crud.SortingOrder;
/**
* Managed bean aggregating the sorting and filtering values for all the
* application's lists. A light-weight bean to always keep in the session with
* minimum impact.
*/
@Name("sortingFilteringBean")
@Scope(ScopeType.SESSION)
public class SortingFilteringBean extends BaseManagedBean {
private static final long serialVersionUID = 1L;
private Sorting applicantProductListSorting;
private Filtering applicantProductListFiltering;
private Sorting homePageSorting;
private Filtering homePageFiltering;
/**
* Creates a new instance of SortingFilteringBean.
*/
public SortingFilteringBean() {
// **********************
// Applicant Product List
// **********************
// Sorting
LinkedHashMap<String, SortingOrder> applicantProductListSortingValues = new LinkedHashMap<String, SortingOrder>();
applicantProductListSortingValues.put("applicantName",
SortingOrder.ASCENDING);
applicantProductListSortingValues.put("applicantEmail",
SortingOrder.ASCENDING);
applicantProductListSortingValues.put("productName",
SortingOrder.ASCENDING);
applicantProductListSortingValues.put("productEmail",
SortingOrder.ASCENDING);
applicantProductListSorting = new Sorting(
applicantProductListSortingValues);
// Filtering
HashMap<String, String> applicantProductListFilteringValues = new HashMap<String, String>();
applicantProductListFilteringValues.put("applicantName", "");
applicantProductListFilteringValues.put("applicantEmail", "");
applicantProductListFilteringValues.put("productName", "");
applicantProductListFilteringValues.put("productEmail", "");
applicantProductListFiltering = new Filtering(
applicantProductListFilteringValues);
// *********
// Home page
// *********
// Sorting
LinkedHashMap<String, SortingOrder> homePageSortingValues = new LinkedHashMap<String, SortingOrder>();
homePageSortingValues.put("productName", SortingOrder.ASCENDING);
homePageSortingValues.put("productId", SortingOrder.ASCENDING);
homePageSortingValues.put("productAtcCode", SortingOrder.UNSORTED);
homePageSortingValues.put("productEmaNumber", SortingOrder.UNSORTED);
homePageSortingValues.put("productOrphan", SortingOrder.UNSORTED);
homePageSortingValues.put("productRap", SortingOrder.UNSORTED);
homePageSortingValues.put("productCorap", SortingOrder.UNSORTED);
homePageSortingValues.put("applicationTypeDescription",
SortingOrder.ASCENDING);
homePageSortingValues.put("applicationId", SortingOrder.ASCENDING);
homePageSortingValues
.put("applicationEmaNumber", SortingOrder.UNSORTED);
homePageSortingValues
.put("piVersionImportDate", SortingOrder.ASCENDING);
homePageSortingValues.put("piVersionId", SortingOrder.ASCENDING);
homePageSorting = new Sorting(homePageSortingValues);
// Filtering
HashMap<String, String> homePageFilteringValues = new HashMap<String, String>();
homePageFilteringValues.put("productName", "");
homePageFilteringValues.put("productAtcCode", "");
homePageFilteringValues.put("productEmaNumber", "");
homePageFilteringValues.put("applicationTypeId", "");
homePageFilteringValues.put("applicationEmaNumber", "");
homePageFilteringValues.put("piVersionImportDate", "");
homePageFiltering = new Filtering(homePageFilteringValues);
}
/**
* @return the applicantProductListFiltering
*/
public Filtering getApplicantProductListFiltering() {
return applicantProductListFiltering;
}
/**
* @param applicantProductListFiltering
* the applicantProductListFiltering to set
*/
public void setApplicantProductListFiltering(
Filtering applicantProductListFiltering) {
this.applicantProductListFiltering = applicantProductListFiltering;
}
/**
* @return the applicantProductListSorting
*/
public Sorting getApplicantProductListSorting() {
return applicantProductListSorting;
}
/**
* @param applicantProductListSorting
* the applicantProductListSorting to set
*/
public void setApplicantProductListSorting(
Sorting applicantProductListSorting) {
this.applicantProductListSorting = applicantProductListSorting;
}
/**
* @return the homePageSorting
*/
public Sorting getHomePageSorting() {
return homePageSorting;
}
/**
* @param homePageSorting
* the homePageSorting to set
*/
public void setHomePageSorting(Sorting homePageSorting) {
this.homePageSorting = homePageSorting;
}
/**
* @return the homePageFiltering
*/
public Filtering getHomePageFiltering() {
return homePageFiltering;
}
/**
* @param homePageFiltering
* the homePageFiltering to set
*/
public void setHomePageFiltering(Filtering homePageFiltering) {
this.homePageFiltering = homePageFiltering;
}
/**
* For convenience to view in the Seam Debug page.
*
* @see java.lang.Object#toString()
*/
@Override
public String toString() {
StringBuilder sb = new StringBuilder("");
sb.append("\n\n");
sb.append("applicantProductListSorting");
sb.append(applicantProductListSorting);
sb.append("\n\n");
sb.append("applicantProductListFiltering");
sb.append(applicantProductListFiltering);
sb.append("\n\n");
sb.append("homePageSorting");
sb.append(homePageSorting);
sb.append("\n\n");
sb.append("homePageFiltering");
sb.append(homePageFiltering);
return sb.toString();
}
}
И это BaseManagedBean, наследующий AbstractMutable.
import java.io.IOException;
import java.io.OutputStream;
import java.util.List;
import javax.faces.application.FacesMessage;
import javax.faces.application.FacesMessage.Severity;
import javax.faces.context.FacesContext;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.lang.ArrayUtils;
import org.jboss.seam.core.AbstractMutable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.myproj.common.exceptions.WebException;
import com.myproj.common.util.FileUtils;
import com.myproj.common.util.StringUtils;
import com.myproj.web.messages.Messages;
public abstract class BaseManagedBean extends AbstractMutable {
private static final Logger logger = LoggerFactory
.getLogger(BaseManagedBean.class);
private FacesContext facesContext;
/**
* Set a message to be displayed for a specific component.
*
* @param resourceBundle
* the resource bundle where the message appears. Either base or
* id may be used.
* @param summaryResourceId
* the id of the resource to be used as summary. For the detail
* of the element, the element to be used will be the same with
* the suffix {@code _detail}.
* @param parameters
* the parameters, in case the string is parameterizable
* @param severity
* the severity of the message
* @param componentId
* the component id for which the message is destined. Note that
* an appropriate JSF {@code <h:message for="myComponentId">} tag
* is required for the to appear, or alternatively a {@code
* <h:messages>} tag.
*/
protected void setMessage(String resourceBundle, String summaryResourceId,
List<Object> parameters, Severity severity, String componentId,
Messages messages) {
FacesContext context = getFacesContext();
FacesMessage message = messages.getMessage(resourceBundle,
summaryResourceId, parameters);
if (severity != null) {
message.setSeverity(severity);
}
context.addMessage(componentId, message);
}
/**
* Copies a byte array to the response output stream with the appropriate
* MIME type and content disposition. The response output stream is closed
* after this method.
*
* @param response
* the HTTP response
* @param bytes
* the data
* @param filename
* the suggested file name for the client
* @param mimeType
* the MIME type; will be overridden if the filename suggests a
* different MIME type
* @throws IllegalArgumentException
* if the data array is <code>null</code>/empty or both filename
* and mimeType are <code>null</code>/empty
*/
protected void printBytesToResponse(HttpServletResponse response,
byte[] bytes, String filename, String mimeType)
throws WebException, IllegalArgumentException {
if (response.isCommitted()) {
throw new WebException("HTTP response is already committed");
}
if (ArrayUtils.isEmpty(bytes)) {
throw new IllegalArgumentException("Data buffer is empty");
}
if (StringUtils.isEmpty(filename) && StringUtils.isEmpty(mimeType)) {
throw new IllegalArgumentException(
"Filename and MIME type are both null/empty");
}
// Set content type (mime type)
String calculatedMimeType = FileUtils.getMimeType(filename);
// not among the known ones
String newMimeType = mimeType;
if (calculatedMimeType == null) {
// given mime type passed
if (mimeType == null) {
// none available put default mime-type
newMimeType = "application/download";
} else {
if ("application/octet-stream".equals(mimeType)) {
// small modification
newMimeType = "application/download";
}
}
} else { // calculated mime type has precedence over given mime type
newMimeType = calculatedMimeType;
}
response.setContentType(newMimeType);
// Set content disposition and other headers
String contentDisposition = "attachment;filename=\"" + filename + "\"";
response.setHeader("Content-Disposition", contentDisposition);
response.setHeader("Expires", "0");
response.setHeader("Cache-Control", "max-age=30");
response.setHeader("Pragma", "public");
// Set content length
response.setContentLength(bytes.length);
// Write bytes to response
OutputStream out = null;
try {
out = response.getOutputStream();
out.write(bytes);
} catch (IOException e) {
throw new WebException("Error writing data to HTTP response", e);
} finally {
try {
out.close();
} catch (Exception e) {
logger.error("Error closing HTTP stream", e);
}
}
}
/**
* Retrieve a session-scoped managed bean.
*
* @param sessionBeanName
* the session-scoped managed bean name
* @return the session-scoped managed bean
*/
protected Object getSessionBean(String sessionBeanName) {
Object sessionScopedBean = FacesContext.getCurrentInstance()
.getExternalContext().getSessionMap().get(sessionBeanName);
if (sessionScopedBean == null) {
throw new IllegalArgumentException("No such object in Session");
} else {
return sessionScopedBean;
}
}
/**
* Set a session-scoped managed bean
*
* @param sessionBeanName
* the session-scoped managed bean name
* @return the session-scoped managed bean
*/
protected boolean setSessionBean(String sessionBeanName, Object sessionBean) {
Object sessionScopedBean = FacesContext.getCurrentInstance()
.getExternalContext().getSessionMap().get(sessionBeanName);
if (sessionScopedBean == null) {
FacesContext.getCurrentInstance().getExternalContext()
.getSessionMap().put(sessionBeanName, sessionBean);
} else {
throw new IllegalArgumentException(
"This session-scoped bean was already initialized");
}
return true;
}
/**
* For testing (enables mock of FacesContext)
*
* @return the faces context
*/
public FacesContext getFacesContext() {
if (facesContext == null) {
return FacesContext.getCurrentInstance();
}
return facesContext;
}
/**
* For testing (enables mocking of FacesContext).
*
* @param aFacesContext
* a - possibly mock - faces context.
*/
public void setFacesContext(FacesContext aFacesContext) {
this.facesContext = aFacesContext;
}
}