Re: getThreadObjectContext with domain parameter - code -

From: Arnaud Garcia (arnau..magemed-87.com)
Date: Mon Sep 14 2009 - 11:03:32 EDT

  • Next message: Andrey Razumovsky: "Re: getThreadObjectContext with domain parameter - code -"

    OK, so I just change by adding new ObjectContext to a Map and set/get
    this Map from the ThreadLocal which is only as the map created one
    time...
    But, I still have the same error sometimes....

    below the new code if you can have a quick look...

    many thanks, for help !

    arnaud
    ---------------------------------------------------------------------------------------
    public class ImagemedBaseContext {

        private static Map<String, ObjectContext> objectContextMap =
    Collections.synchronizedMap(new HashMap<String, ObjectContext>());
        private static final Logger log =
    Logger.getLogger(ImagemedBaseContext.class.getName());
        protected static final ThreadLocal<Map<String, ObjectContext>>
    threadObjectContext = new ThreadLocal<Map<String, ObjectContext>>();

        public static ObjectContext getThreadObjectContext(String domain)
    throws IllegalStateException {
            ObjectContext context = threadObjectContext.get().get(domain);
            if (context == null) {
                throw new IllegalStateException("Current thread has no
    bound ObjectContext.");
            } else {
                log.fine("[GET OBJECT CONTEXT] ObjectContext (" +
    context.hashCode() + ") for domain=" + domain + " to ThreadLocal (" +
    threadObjectContext.hashCode() + ")");
            }
            return context;

        }

        public static void bindThreadObjectContext(ObjectContext context,
    String domain) {
            objectContextMap.put(domain, context);
            threadObjectContext.set(objectContextMap);
            if (context == null) {
                log.fine("[UNBIND OBJECT CONTEXT]");
            } else {
                log.fine("[BIND OBJECT CONTEXT] ObjectContext (" +
    context.hashCode() + ") for domain=" + domain + " to ThreadLocal (" +
    threadObjectContext.hashCode() + ")");
            }
        }
    }
    ---------------------------------------------------------------------------------------
    ---------------------------------------------------------------------------------------

    2009/9/11 Andrey Razumovsky <razumovsky.andre..mail.com>:
    > Well, looking at your code, I think you should not create new ThreadLocal
    > each time request is received (in bindThreadObjectContext). You old bindings
    > just get erased
    >
    > 2009/9/11 Arnaud Garcia <arnau..magemed-87.com>
    >
    >> Hello,
    >>
    >> I created a filter to use multi domains in a webapp, but I still have
    >> a problem...., I don't find the bug... if someone can help me to fix
    >> it !
    >> (I think that some people will also need this kind of filter, so this
    >> is  why I post the complete code)
    >>
    >> So, the last bug is that sometimes the ObjectContext cannot be
    >> retrieved.....
    >> java.lang.IllegalStateException: Current thread has no bound ObjectContext.
    >>     at
    >> com.imagemed.cayenne.ImagemedBaseContext.getThreadObjectContext(ImagemedBaseContext.java:39)
    >>
    >> thanks
    >>
    >> Arnaud
    >>
    >>
    >> To use the filter:
    >> ----------------------
    >>
    >> =>Declare the filter in web.xml as usual :
    >>
    >>  <filter>
    >>        <filter-name>Imagemed.Cayenne</filter-name>
    >>
    >>  <filter-class>com.imagemed.cayenne.WebApplicationMultiDomainsContextFilter</filter-class>
    >>    </filter>
    >>
    >>    <filter-mapping>
    >>        <filter-name>Imagemed.Cayenne</filter-name>
    >>        <url-pattern>/*</url-pattern>
    >>    </filter-mapping>
    >>
    >> Then in your DAO you retrieve your contexts from your session like this:
    >>
    >>  ctxtForDomain1 = (DataContext)
    >> ImagemedBaseContext.getThreadObjectContext("DOMAIN1");
    >>  ctxtForDomain2 = (DataContext)
    >> ImagemedBaseContext.getThreadObjectContext("DOMAIN2");
    >> ....
    >>
    >> Below you will find the complete code.....
    >>
    >>
    >> ------------------------------ THE FILTER -------------------------
    >> package com.imagemed.cayenne;
    >>
    >> import java.io.IOException;
    >> import java.util.ArrayList;
    >> import java.util.Collection;
    >> import java.util.logging.Logger;
    >> import javax.servlet.Filter;
    >> import javax.servlet.FilterChain;
    >> import javax.servlet.FilterConfig;
    >> import javax.servlet.ServletException;
    >> import javax.servlet.ServletRequest;
    >> import javax.servlet.ServletResponse;
    >> import javax.servlet.http.HttpServletRequest;
    >> import javax.servlet.http.HttpSession;
    >> import org.apache.cayenne.access.DataContext;
    >> import org.apache.cayenne.access.DataDomain;
    >> import org.apache.cayenne.conf.Configuration;
    >> import org.apache.cayenne.conf.ServletUtil;
    >>
    >>
    >> public class WebApplicationMultiDomainsContextFilter implements Filter {
    >>
    >>    public static final String DATA_CONTEXT_KEY = "cayenne.datacontext";
    >>    private ArrayList<String> domains = new ArrayList<String>();
    >>    private static final Logger log =
    >> Logger.getLogger(WebApplicationMultiDomainsContextFilter.class.getName());
    >>
    >>    public static DataContext getSessionContext(HttpSession session,
    >> String domain) {
    >>        synchronized (session) {
    >>            String dataCtxKey = DATA_CONTEXT_KEY + "." + domain;
    >>            DataContext ctxt = (DataContext)
    >> session.getAttribute(dataCtxKey);
    >>
    >>            if (ctxt == null) {
    >>                ctxt = DataContext.createDataContext(domain);
    >>                session.setAttribute(dataCtxKey, ctxt);
    >>            } else {
    >>                log.fine("[RETRIEVE] DataContext (" + ctxt.hashCode()
    >> + ") retrieved from session key=" + dataCtxKey);
    >>            }
    >>
    >>            return ctxt;
    >>        }
    >>    }
    >>
    >>    public void destroy() {
    >>    }
    >>
    >>    public void doFilter(ServletRequest request, ServletResponse
    >> response, FilterChain chain) throws IOException, ServletException {
    >>        boolean reset = false;
    >>
    >>        if (request instanceof HttpServletRequest) {
    >>            reset = true;
    >>
    >>            HttpSession session = ((HttpServletRequest)
    >> request).getSession(true);
    >>            for (String domain : domains) {
    >>                DataContext context = getSessionContext(session, domain);
    >>                ImagemedBaseContext.bindThreadObjectContext(context,
    >> domain);
    >>            }
    >>        }
    >>
    >>        try {
    >>            chain.doFilter(request, response);
    >>        } finally {
    >>            if (reset) {
    >>                for (String domain : domains) {
    >>                    ImagemedBaseContext.bindThreadObjectContext(null,
    >> domain);
    >>                }
    >>            }
    >>        }
    >>    }
    >>
    >>    public void init(FilterConfig filterConfig) throws ServletException {
    >>
    >>  ServletUtil.initializeSharedConfiguration(filterConfig.getServletContext());
    >>        Collection<DataDomain> collDomains =
    >> Configuration.getSharedConfiguration().getDomains();
    >>        for (DataDomain dataDomain : collDomains) {
    >>            domains.add(dataDomain.getName());
    >>            log.info("[FOUND DOMAIN] name=" + dataDomain.getName());
    >>        }
    >>    }
    >> }
    >>
    >> ------------------------------ ImagemedBaseContext ------------------------
    >> package com.imagemed.cayenne;
    >>
    >> import java.util.Collections;
    >> import java.util.HashMap;
    >> import java.util.Map;
    >> import java.util.logging.Logger;
    >> import org.apache.cayenne.ObjectContext;
    >>
    >> public class ImagemedBaseContext {
    >>
    >>    private static Map<String, ThreadLocal> threadLocalMap =
    >> Collections.synchronizedMap(new HashMap<String, ThreadLocal>());
    >>    private static final Logger log =
    >> Logger.getLogger(ImagemedBaseContext.class.getName());
    >>
    >>    public static ObjectContext getThreadObjectContext(String domain)
    >> throws IllegalStateException {
    >>        ThreadLocal<ObjectContext> threadObjectContext =
    >> threadLocalMap.get(domain);
    >>        ObjectContext context = threadObjectContext.get();
    >>
    >>        if (context == null) {
    >>            throw new IllegalStateException("Current thread has no
    >> bound ObjectContext.");       <============================ ligne 39
    >>        } else {
    >>            log.fine("[GET OBJECT CONTEXT] ObjectContext (" +
    >> context.hashCode() + ") for domain=" + domain + " to ThreadLocal (" +
    >> threadObjectContext.hashCode() + ")");
    >>        }
    >>        return context;
    >>
    >>    }
    >>
    >>    public static void bindThreadObjectContext(ObjectContext context,
    >> String domain) {
    >>        ThreadLocal<ObjectContext> threadObjectContext = new
    >> ThreadLocal<ObjectContext>();
    >>        threadObjectContext.set(context);
    >>        threadLocalMap.put(domain, threadObjectContext);
    >>        if (context == null) {
    >>            log.fine("[UNBIND OBJECT CONTEXT]");
    >>        } else {
    >>            log.fine("[BIND OBJECT CONTEXT] ObjectContext (" +
    >> context.hashCode() + ") for domain=" + domain + " to ThreadLocal (" +
    >> threadObjectContext.hashCode() + ")");
    >>        }
    >>    }
    >> }
    >>
    >
    >
    >
    > --
    > Andrey
    >



    This archive was generated by hypermail 2.0.0 : Mon Sep 14 2009 - 11:04:15 EDT