org.sakaiproject.lessonbuildertool.service.ForumEntity.java Source code

Java tutorial

Introduction

Here is the source code for org.sakaiproject.lessonbuildertool.service.ForumEntity.java

Source

/**********************************************************************************
 * $URL: $
 * $Id: $
 ***********************************************************************************
 *
 * Author: Charles Hedrick, hedrick@rutgers.edu
 *
 * Copyright (c) 2010 Rutgers, the State University of New Jersey
 *
 * Licensed under the Educational Community 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.opensource.org/licenses/ECL-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.sakaiproject.lessonbuildertool.service;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Map;
import java.util.Collection;
import java.util.List;
import java.util.Set;
import java.util.SortedSet;
import java.util.HashSet;
import java.util.TreeSet;
import java.util.Comparator;
import java.util.Date;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.commons.lang.StringEscapeUtils;

import org.sakaiproject.lessonbuildertool.service.LessonSubmission;
import org.sakaiproject.lessonbuildertool.tool.beans.SimplePageBean;
import org.sakaiproject.lessonbuildertool.tool.beans.SimplePageBean.UrlItem;
import org.sakaiproject.api.app.messageforums.BaseForum;
import org.sakaiproject.api.app.messageforums.DiscussionForum;
import org.sakaiproject.api.app.messageforums.DiscussionTopic;
import org.sakaiproject.api.app.messageforums.Topic;
import org.sakaiproject.api.app.messageforums.MessageForumsForumManager;
import org.sakaiproject.api.app.messageforums.ui.DiscussionForumManager;
import org.sakaiproject.api.app.messageforums.MessageForumsMessageManager;
import org.sakaiproject.api.app.messageforums.PermissionLevelManager;
import org.sakaiproject.api.app.messageforums.MessageForumsTypeManager;
import org.sakaiproject.api.app.messageforums.AreaManager;
import org.sakaiproject.api.app.messageforums.Attachment;
import org.sakaiproject.api.app.messageforums.DBMembershipItem;
import org.sakaiproject.api.app.messageforums.PermissionLevel;
import org.sakaiproject.api.app.messageforums.PermissionsMask;
import org.sakaiproject.api.app.messageforums.ui.UIPermissionsManager;
import org.sakaiproject.component.app.messageforums.MembershipItem;

import org.sakaiproject.component.cover.ComponentManager;
import org.sakaiproject.tool.cover.ToolManager;
import org.sakaiproject.site.api.ToolConfiguration;
import org.sakaiproject.site.api.Group;
import org.sakaiproject.site.api.Site;
import org.sakaiproject.authz.api.Role;
import org.sakaiproject.site.cover.SiteService;
import org.sakaiproject.authz.api.AuthzGroupService;
import org.sakaiproject.id.cover.IdManager;
import org.sakaiproject.component.cover.ServerConfigurationService;
import org.sakaiproject.db.cover.SqlService;

import org.sakaiproject.memory.api.Cache;
import org.sakaiproject.memory.api.CacheRefresher;
import org.sakaiproject.memory.api.MemoryService;

import org.sakaiproject.util.FormattedText;
import java.net.URLEncoder;

import uk.org.ponder.messageutil.MessageLocator;

import org.sakaiproject.lessonbuildertool.model.SimplePageToolDao;
import org.springframework.orm.hibernate3.support.HibernateDaoSupport;
import org.springframework.orm.hibernate3.HibernateTemplate;
import org.springframework.orm.hibernate3.HibernateCallback;
import org.hibernate.SessionFactory;
import org.hibernate.Session;
import org.hibernate.Transaction;

/**
 * Interface to Message Forums, the forum that comes with Sakai
 *
 * @author Charles Hedrick <hedrick@rutgers.edu>
 * 
 */

// NOTE: almost no other class should import this. We want to be able
// to support both forums and jforum. So typically there will be a 
// forumEntity, but it's injected, and it can be either forum and jforum.
// Hence it has to be declared LessonEntity. That leads to a lot of
// declarations like LessonEntity forumEntity.  In this case forumEntity
// means either a ForumEntity or a JForumEntity. We can't just call the
// variables lessonEntity because the same module will probably have an
// injected class to handle tests and quizes as well. That will eventually
// be converted to be a LessonEntity.

// this monstrosity has to do hibernate directly to avoid the dreaded 
// multiple objects with the same ID error. I do merge rather than save.
// unfortunately the normal API does saveOrUpdate.
//  This is complicated by the fact that we sessionFactory is set only
// when the main bean is set up. But setGroups is only called from 
// instances of this class that aren't the bean. Hence in the bean
// we save a copy of the session factory and then set it in the
// instance when we need it.

public class ForumEntity extends HibernateDaoSupport implements LessonEntity, ForumInterface {

    private static Log log = LogFactory.getLog(ForumEntity.class);

    private static Cache topicCache = null; // topicid => grouplist
    protected static final int DEFAULT_EXPIRATION = 10 * 60;
    private static SessionFactory sessionFactory = null;

    static MessageForumsForumManager forumManager = (MessageForumsForumManager) ComponentManager
            .get("org.sakaiproject.api.app.messageforums.MessageForumsForumManager");
    static MessageForumsMessageManager messageManager = (MessageForumsMessageManager) ComponentManager
            .get("org.sakaiproject.api.app.messageforums.MessageForumsMessageManager");
    static PermissionLevelManager permissionLevelManager = (PermissionLevelManager) ComponentManager
            .get("org.sakaiproject.api.app.messageforums.PermissionLevelManager");
    static UIPermissionsManager uiPermissionsManager = (UIPermissionsManager) ComponentManager
            .get("org.sakaiproject.api.app.messageforums.ui.UIPermissionsManager");
    static DiscussionForumManager discussionForumManager = (DiscussionForumManager) ComponentManager
            .get("org.sakaiproject.api.app.messageforums.ui.DiscussionForumManager");
    static AreaManager areaManager = (AreaManager) ComponentManager
            .get("org.sakaiproject.api.app.messageforums.AreaManager");
    static MessageForumsTypeManager typeManager = (MessageForumsTypeManager) ComponentManager
            .get("org.sakaiproject.api.app.messageforums.MessageForumsTypeManager");

    private LessonEntity nextEntity = null;

    public void setNextEntity(LessonEntity e) {
        nextEntity = e;
    }

    public LessonEntity getNextEntity() {
        return nextEntity;
    }

    static MessageLocator messageLocator = null;

    public void setMessageLocator(MessageLocator m) {
        messageLocator = m;
    }

    static MemoryService memoryService = null;

    public void setMemoryService(MemoryService m) {
        memoryService = m;
    }

    static AuthzGroupService authzGroupService = null;

    public void setAuthzGroupService(AuthzGroupService service) {
        authzGroupService = service;
    }

    private static SimplePageToolDao simplePageToolDao;

    public void setSimplePageToolDao(Object dao) {
        //   System.out.println("set dao " + dao);
        simplePageToolDao = (SimplePageToolDao) dao;
    }

    private static HibernateTemplate hibernateTemplate = null;

    public void init() {
        //   topicCache = memoryService
        //       .newCache("org.sakaiproject.lessonbuildertool.service.ForumEntity.cache");
        sessionFactory = getSessionFactory();
    }

    protected void initDao() throws Exception {
        super.initDao();
        log.info("initDao template " + getHibernateTemplate());
        hibernateTemplate = getHibernateTemplate();
    }

    public void destroy() {
        //   topicCache.destroy();
        //   topicCache = null;

        log.info("destroy()");
    }

    // to create bean. the bean is used only to call the pseudo-static
    // methods such as getEntitiesInSite. So type, id, etc are left uninitialized

    protected ForumEntity() {
    }

    protected ForumEntity(int type, Long id, int level) {
        this.type = type;
        this.id = id;
        this.level = level;
    }

    public String getToolId() {
        return "sakai.forums";
    }

    // the underlying object, something Sakaiish
    protected Long id;
    protected int type;
    protected int level;
    // not required fields. If we need to look up
    // the actual objects, lets us cache them
    protected Topic topic = null;
    protected BaseForum forum = null;

    // type of the underlying object
    public int getType() {
        return type;
    }

    public int getLevel() {
        return level;
    }

    public int getTypeOfGrade() {
        return 1;
    }

    public boolean isUsable() {
        return true;
    }

    public String getReference() {
        if (type == TYPE_FORUM_TOPIC)
            return "/" + FORUM_TOPIC + "/" + id;
        else
            return "/" + FORUM_FORUM + "/" + id;
    }

    public class ForumBySortIndexAscAndCreatedDateDesc implements Comparator<BaseForum> {

        public int compare(BaseForum forum, BaseForum otherForum) {
            if (forum != null && otherForum != null) {
                Integer index1 = forum.getSortIndex();
                Integer index2 = otherForum.getSortIndex();
                if (index1.intValue() != index2.intValue())
                    return index1.intValue() - index2.intValue();
                Date date1 = forum.getCreated();
                Date date2 = otherForum.getCreated();
                int rval = date2.compareTo(date1);
                if (rval == 0) {
                    return otherForum.getId().compareTo(forum.getId());
                } else {
                    return rval;
                }
            }
            return -1;
        }

    }

    public List<LessonEntity> getEntitiesInSite() {
        return getEntitiesInSite(null);
    }

    // find topics in site, but organized by forum
    public List<LessonEntity> getEntitiesInSite(SimplePageBean bean) {

        List<LessonEntity> ret = new ArrayList<LessonEntity>();

        // LSNBLDR-21. If the tool is not in the current site we shouldn't query
        // for topics owned by the tool.
        Site site = null;
        try {
            site = SiteService.getSite(ToolManager.getCurrentPlacement().getContext());
        } catch (Exception impossible) {
            return ret;
        }

        ToolConfiguration tool = site.getToolForCommonId("sakai.forums");

        if (tool == null) {

            // Forums is not in this site. Move on to the next provider.

            if (nextEntity != null)
                ret.addAll(nextEntity.getEntitiesInSite());

            return ret;
        }

        //ForumEntity e = new ForumEntity(TYPE_FORUM_TOPIC, 3L, 2);

        //e.setGroups(Arrays.asList("1c24287b-b880-43da-8cdd-c6cdc1249c5c", "75184424-853e-4dd4-9e92-980c851f0580"));
        //e.setGroups(Arrays.asList("75184424-853e-4dd4-9e92-980c851f0580"));

        SortedSet<DiscussionForum> forums = new TreeSet<DiscussionForum>(
                new ForumBySortIndexAscAndCreatedDateDesc());
        for (DiscussionForum forum : forumManager.getForumsForMainPage())
            forums.add(forum);

        // security. assume this is only used in places where it's OK, so skip security checks
        for (DiscussionForum forum : forums) {
            if (!forum.getDraft()) {
                ForumEntity entity = new ForumEntity(TYPE_FORUM_FORUM, forum.getId(), 1);
                entity.forum = forum;
                ret.add(entity);
                for (Object o : forum.getTopicsSet()) {
                    DiscussionTopic topic = (DiscussionTopic) o;
                    if (topic.getDraft().equals(Boolean.FALSE)) {
                        entity = new ForumEntity(TYPE_FORUM_TOPIC, topic.getId(), 2);
                        entity.topic = topic;
                        ret.add(entity);
                    }
                }
            }
        }

        if (nextEntity != null)
            ret.addAll(nextEntity.getEntitiesInSite(bean));

        return ret;
    }

    public LessonEntity getEntity(String ref, SimplePageBean o) {
        return getEntity(ref);
    }

    public LessonEntity getEntity(String ref) {
        int i = ref.indexOf("/", 1);
        if (i < 0)
            return null;
        String typeString = ref.substring(1, i);
        String idString = ref.substring(i + 1);
        Long id = 0L;
        if (typeString.equals(FORUM_TOPIC) || typeString.equals(FORUM_FORUM)) {
            try {
                id = Long.parseLong(idString);
            } catch (Exception ignore) {
                return null;
            }
        }

        // note: I'm returning the minimal structures, not those with
        // topics and postings attached
        if (typeString.equals(FORUM_TOPIC)) {
            return new ForumEntity(TYPE_FORUM_TOPIC, id, 2);
        } else if (typeString.equals(FORUM_FORUM)) {
            return new ForumEntity(TYPE_FORUM_FORUM, id, 1);
        } else if (nextEntity != null) {
            return nextEntity.getEntity(ref);
        } else
            return null;
    }

    public Topic getTopicById(boolean flag, Long id) {
        try {
            Topic topic = forumManager.getTopicById(flag, id);
            if (topic == null)
                return null;
            // the purpose of this is to trigger a null pointer error if the forum doesn't exist
            // if you delete the forum, it doesn't delete the topics, but showing a topic without
            // its forum causes confusion. The /direct code does the following, so it will give
            // an error. 
            if (topic.getOpenForum().getArea() == null)
                return null;
            return topic;
        } catch (Exception e) {
            return null;
        }
    }

    public BaseForum getForumById(boolean flag, Long id) {
        try {
            return forumManager.getForumById(flag, id);
        } catch (Exception e) {
            return null;
        }
    }

    // properties of entities
    public String getTitle() {

        if (type == TYPE_FORUM_TOPIC) {
            if (topic == null)
                topic = getTopicById(true, id);
            if (topic == null)
                return null;
            return topic.getTitle();
        } else {
            if (forum == null)
                forum = getForumById(true, id);
            if (forum == null)
                return null;
            return forum.getTitle();
        }
    }

    public String getUrl() {

        if (type == TYPE_FORUM_TOPIC) {
            if (topic == null)
                topic = getTopicById(true, id);
            if (topic == null)
                return "javascript:alert('" + messageLocator.getMessage("simplepage.forumdeleted") + "')";
        } else {
            if (forum == null)
                forum = getForumById(true, id);
            if (forum == null)
                return "javascript:alert('" + messageLocator.getMessage("simplepage.forumdeleted") + "')";
        }

        Site site = null;
        try {
            site = SiteService.getSite(ToolManager.getCurrentPlacement().getContext());
        } catch (Exception impossible) {
            return null;
        }
        ToolConfiguration tool = site.getToolForCommonId("sakai.forums");

        // LSNBLDR-21. If the tool is not in the current site we shouldn't return a url
        if (tool == null) {
            return null;
        }

        String placement = tool.getId();

        if (type == TYPE_FORUM_TOPIC)
            // if /direct doesn't work, but that was only in one beta release
            //  return "/messageforums-tool/jsp/discussionForum/message/dfAllMessagesDirect.jsf?topicId=" + id + "&placementId=" + placement;
            return "/direct/forum_topic/" + id;
        else
            return "/direct/forum/" + id;
    }

    // I don't think they have this
    public Date getDueDate() {
        return null;
    }

    // The msgcntr permissions model is completely undocumented. Here's a cheat sheet:

    // DBMembershipItem has 
    //    type, which is ALL, ROLE, GROUP or USER
    //    name which is the specific rolename, groupname or username
    //    permissionlevelname which is "Owner", "Contributor", "None", etc.
    //       In addition there is a bitmask that says which specific permissions
    //       are present, but we always use the default permissions for each level.

    // Here's typical code. First we create a permissionlevel with a given bitmask.
    // This permissionlevel controls the detailed permissions. As noted, we always 
    // default levels. Unfortunately we have to create a new copy of the level
    // for every entry. Some fo their internal code uses common permissionlevels, but
    // if you don't have a separate level object and database entry for each membershipitem,
    // things get very confused.
    //     PermissionLevel contributorLevel = permissionLevelManager.
    //         createPermissionLevel("Contributor",  typeManager.getContributorLevelType(), contributorMask);
    //    permissionLevelManager.savePermissionLevel(contributorLevel);

    // Now we create the actual entry. Note that this one says members of the specified
    // group are contributors. Then it sets the default contributor bitmask. You can call
    // a permission contributor but set any bits you want. However we're going to assume
    // that people pick a name that represents what they want, and make minimal changes.
    //     DBMembershipItem membershipItem = permissionLevelManager.
    //       createDBMembershipItem(groupName, "Contributor", MembershipItem.TYPE_GROUP);
    //     membershipItem.setPermissionLevel(contributorLevel);
    //     permissionLevelManager.saveDBMembershipItem(membershipItem);   

    // How we use it:

    // ACCESS CONTROL:

    // Our model is fairly simple. When we control access, Owner is the maintain role, and 
    //    contributor is the group we control. Everything else is set to none.
    // When you decontrol something, we make Owner the maintain role
    //    and Contributor all the other roles. Once we support groups, we'll put back
    //    saved group access, but we won't try to put back anything else.
    // The tool code makes sure that items are added for all roles, so we don't have to add entries
    //    in most cases, just change their permission levels.

    // GROUP ACCESS WHEN WE AREN'T CONTROLLING:

    // Now, our group management code, which should only be used when we're not controlling:
    // Getgroups returns which groups are contributor.
    //    If you want something more complex, you'll have to do it in the tool, but if there
    //       are any groups with contributor, we'll only let students in those groups access
    //       through our tool.
    // Setgroups with a non-null list: we set all contributor entries to none, and then set the
    //    specified groups to contribtor. By only handling groups, we avoid interfering with
    //    anything you might do in the tool. But the moment you use access control, we take
    //    over. Sorry. Once we've done that you could go back into the tool and hack, but I
    //    don't recommend that.
    // Setgroups with a null list: we set all contributor entries to none, and then set all roles
    //    other than maintain to contributor.
    // It may be safer to do changes in the tool.

    // the following methods all take references. So they're in effect static.
    // They ignore the entity from which they're called.
    // The reason for not making them a normal method is that many of the
    // implementations seem to let you set access control and find submissions
    // from a reference, without needing the actual object. So doing it this
    // way could save some database activity

    PermissionsMask noneMask = null;
    PermissionsMask contributorMask = null;
    PermissionsMask ownerMask = null;

    private void setMasks() {
        if (noneMask == null) {
            noneMask = new PermissionsMask();
            noneMask.put(PermissionLevel.NEW_FORUM, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.NEW_TOPIC, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.NEW_RESPONSE, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.NEW_RESPONSE_TO_RESPONSE, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.MOVE_POSTING, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.CHANGE_SETTINGS, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.POST_TO_GRADEBOOK, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.READ, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.MARK_AS_READ, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.MODERATE_POSTINGS, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.DELETE_OWN, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.DELETE_ANY, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.REVISE_OWN, Boolean.valueOf(false));
            noneMask.put(PermissionLevel.REVISE_ANY, Boolean.valueOf(false));
        }
        if (contributorMask == null) {
            contributorMask = new PermissionsMask();
            contributorMask.put(PermissionLevel.NEW_FORUM, Boolean.valueOf(false));
            contributorMask.put(PermissionLevel.NEW_TOPIC, Boolean.valueOf(false));
            contributorMask.put(PermissionLevel.NEW_RESPONSE, Boolean.valueOf(true));
            contributorMask.put(PermissionLevel.NEW_RESPONSE_TO_RESPONSE, Boolean.valueOf(true));
            contributorMask.put(PermissionLevel.MOVE_POSTING, Boolean.valueOf(false));
            contributorMask.put(PermissionLevel.CHANGE_SETTINGS, Boolean.valueOf(false));
            contributorMask.put(PermissionLevel.POST_TO_GRADEBOOK, Boolean.valueOf(false));
            contributorMask.put(PermissionLevel.READ, Boolean.valueOf(true));
            contributorMask.put(PermissionLevel.MARK_AS_READ, Boolean.valueOf(true));
            contributorMask.put(PermissionLevel.MODERATE_POSTINGS, Boolean.valueOf(false));
            contributorMask.put(PermissionLevel.DELETE_OWN, Boolean.valueOf(false));
            contributorMask.put(PermissionLevel.DELETE_ANY, Boolean.valueOf(false));
            contributorMask.put(PermissionLevel.REVISE_OWN, Boolean.valueOf(false));
            contributorMask.put(PermissionLevel.REVISE_ANY, Boolean.valueOf(false));
        }
        if (ownerMask == null) {
            ownerMask = new PermissionsMask();
            ownerMask.put(PermissionLevel.NEW_FORUM, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.NEW_TOPIC, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.NEW_RESPONSE, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.NEW_RESPONSE_TO_RESPONSE, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.MOVE_POSTING, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.CHANGE_SETTINGS, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.POST_TO_GRADEBOOK, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.READ, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.MARK_AS_READ, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.MODERATE_POSTINGS, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.DELETE_OWN, Boolean.valueOf(false));
            ownerMask.put(PermissionLevel.DELETE_ANY, Boolean.valueOf(true));
            ownerMask.put(PermissionLevel.REVISE_OWN, Boolean.valueOf(false));
            ownerMask.put(PermissionLevel.REVISE_ANY, Boolean.valueOf(true));
        }
    }

    // access control
    // seems not to be used anymore
    public boolean addEntityControl(String siteId, String groupId) throws IOException {

        if (type != TYPE_FORUM_TOPIC)
            return false;

        setMasks();

        if (topic == null)
            topic = getTopicById(true, id);
        if (topic == null)
            return false;

        Set<DBMembershipItem> oldMembershipItemSet = uiPermissionsManager.getTopicItemsSet((DiscussionTopic) topic);

        Set membershipItemSet = new HashSet();

        String groupName = null;
        String maintainRole = null;
        try {
            Site site = SiteService.getSite(ToolManager.getCurrentPlacement().getContext());
            groupName = site.getGroup(groupId).getTitle();
            maintainRole = authzGroupService.getAuthzGroup("/site/" + site.getId()).getMaintainRole();
        } catch (Exception e) {
            System.out.println("Unable to get site info for AddEntityControl " + e);
        }

        PermissionLevel ownerLevel = permissionLevelManager.createPermissionLevel("Owner",
                typeManager.getOwnerLevelType(), ownerMask);
        permissionLevelManager.savePermissionLevel(ownerLevel);

        PermissionLevel contributorLevel = permissionLevelManager.createPermissionLevel("Contributor",
                typeManager.getContributorLevelType(), contributorMask);
        permissionLevelManager.savePermissionLevel(contributorLevel);

        DBMembershipItem membershipItem = permissionLevelManager.createDBMembershipItem(groupName, "Contributor",
                MembershipItem.TYPE_GROUP);
        membershipItem.setPermissionLevel(contributorLevel);
        permissionLevelManager.saveDBMembershipItem(membershipItem);

        membershipItemSet.add(membershipItem);

        membershipItem = permissionLevelManager.createDBMembershipItem(maintainRole, "Owner",
                MembershipItem.TYPE_ROLE);
        membershipItem.setPermissionLevel(ownerLevel);
        permissionLevelManager.saveDBMembershipItem(membershipItem);

        membershipItemSet.add(membershipItem);

        // now change any existing ones into null
        for (DBMembershipItem item : oldMembershipItemSet) {
            if (!(maintainRole.equals(item.getName()) && item.getType().equals(MembershipItem.TYPE_ROLE)
                    || groupName.equals(item.getName()) && item.getType().equals(MembershipItem.TYPE_GROUP))) {
                PermissionLevel noneLevel = permissionLevelManager.createPermissionLevel("None",
                        typeManager.getNoneLevelType(), noneMask);
                permissionLevelManager.savePermissionLevel(noneLevel);

                membershipItem = permissionLevelManager.createDBMembershipItem(item.getName(), "None",
                        item.getType());
                membershipItem.setPermissionLevel(noneLevel);
                permissionLevelManager.saveDBMembershipItem(membershipItem);
                membershipItemSet.add(membershipItem);
            }
        }

        permissionLevelManager.deleteMembershipItems(oldMembershipItemSet);

        topic.setMembershipItemSet(membershipItemSet);
        discussionForumManager.saveTopic((DiscussionTopic) topic);

        return true;
    };

    // seems not to be used anymore
    public boolean removeEntityControl(String siteId, String groupId) throws IOException {

        if (type != TYPE_FORUM_TOPIC)
            return false;

        setMasks();

        if (topic == null)
            topic = getTopicById(true, id);
        if (topic == null)
            return false;

        Set<DBMembershipItem> oldMembershipItemSet = uiPermissionsManager.getTopicItemsSet((DiscussionTopic) topic);

        Set membershipItemSet = new HashSet();

        String groupName = null;
        String maintainRole = null;
        try {
            Site site = SiteService.getSite(ToolManager.getCurrentPlacement().getContext());
            groupName = site.getGroup(groupId).getTitle();
            maintainRole = authzGroupService.getAuthzGroup("/site/" + site.getId()).getMaintainRole();
        } catch (Exception e) {
            System.out.println("Unable to get site info for AddEntityControl " + e);
        }

        PermissionLevel ownerLevel = permissionLevelManager.createPermissionLevel("Owner",
                typeManager.getOwnerLevelType(), ownerMask);
        permissionLevelManager.savePermissionLevel(ownerLevel);

        DBMembershipItem membershipItem = permissionLevelManager.createDBMembershipItem(maintainRole, "Owner",
                MembershipItem.TYPE_ROLE);
        membershipItem.setPermissionLevel(ownerLevel);
        permissionLevelManager.saveDBMembershipItem(membershipItem);

        membershipItemSet.add(membershipItem);

        // now change any existing ones into null
        for (DBMembershipItem item : oldMembershipItemSet) {
            if (item.getType().equals(MembershipItem.TYPE_ROLE)) {
                if (!maintainRole.equals(item.getName())) { // that was done above, other roles contributor
                    PermissionLevel contributorLevel = permissionLevelManager.createPermissionLevel("Contributor",
                            typeManager.getContributorLevelType(), contributorMask);
                    permissionLevelManager.savePermissionLevel(contributorLevel);

                    membershipItem = permissionLevelManager.createDBMembershipItem(item.getName(), "Contributor",
                            item.getType());
                    membershipItem.setPermissionLevel(contributorLevel);
                    permissionLevelManager.saveDBMembershipItem(membershipItem);
                    membershipItemSet.add(membershipItem);
                }
            } else { // everything else off
                PermissionLevel noneLevel = permissionLevelManager.createPermissionLevel("None",
                        typeManager.getNoneLevelType(), noneMask);
                permissionLevelManager.savePermissionLevel(noneLevel);

                membershipItem = permissionLevelManager.createDBMembershipItem(item.getName(), "None",
                        item.getType());
                membershipItem.setPermissionLevel(noneLevel);
                permissionLevelManager.saveDBMembershipItem(membershipItem);
                membershipItemSet.add(membershipItem);
            }
        }

        permissionLevelManager.deleteMembershipItems(oldMembershipItemSet);

        topic.setMembershipItemSet(membershipItemSet);
        discussionForumManager.saveTopic((DiscussionTopic) topic);

        return true;
    };

    // submission
    // do we need the data from submission?
    //  not for the moment. If a posting is required, we just check whether one
    //  has been done. While you can grade submissions, grading is done manually
    //  later. It's unlikely that faculty will want to test on those grades
    public boolean needSubmission() {
        return false;
    }

    public LessonSubmission getSubmission(String user) {
        return null; // not used
    }

    public int getSubmissionCount(String user) {
        if (type == TYPE_FORUM_TOPIC)
            return messageManager.findAuhtoredMessageCountByTopicIdByUserId(id, user);
        else
            return messageManager.findAuthoredStatsForStudentByForumId(user, id).size();
    }

    // URL to create a new item. Normally called from the generic entity, not a specific one                                                 
    // can't be null                                                                                                                         
    public List<UrlItem> createNewUrls(SimplePageBean bean) {
        ArrayList<UrlItem> list = new ArrayList<UrlItem>();
        String tool = bean.getCurrentTool("sakai.forums");
        if (tool != null) {
            tool = ServerConfigurationService.getToolUrl() + "/" + tool + "/discussionForum/forumsOnly/dfForums";
            list.add(new UrlItem(tool, messageLocator.getMessage("simplepage.create_forums")));
        }
        if (nextEntity != null)
            list.addAll(nextEntity.createNewUrls(bean));
        return list;
    }

    // URL to edit an existing entity.                                                                                                       
    // Can be null if we can't get one or it isn't needed                                                                                    
    public String editItemUrl(SimplePageBean bean) {
        return getUrl();
    }

    // for most entities editItem is enough, however tests allow separate editing of                                                         
    // contents and settings. This will be null except in that situation                                                                     
    public String editItemSettingsUrl(SimplePageBean bean) {
        return null;
    }

    //   aaa/bb/../cc ==>
    //   aaa/cc
    public String removeDotDot(String s) {
        while (true) {
            int i = s.indexOf("/../");
            if (i < 1)
                return s;
            int j = s.lastIndexOf("/", i - 1);
            if (j < 0)
                j = 0;
            else
                j = j + 1;
            s = s.substring(0, j) + s.substring(i + 4);
        }
    }

    // returns SakaiId of thing just created
    public String importObject(String title, String topicTitle, String text, boolean texthtml, String base,
            String baseDir, String siteId, List<String> attachmentHrefs, boolean hide) {

        DiscussionForum ourForum = null;
        DiscussionTopic ourTopic = null;

        int forumtry = 0;
        int topictry = 0;

        for (;;) {

            ourForum = null;

            SortedSet<DiscussionForum> forums = new TreeSet<DiscussionForum>(
                    new ForumBySortIndexAscAndCreatedDateDesc());
            for (DiscussionForum forum : discussionForumManager.getForumsForMainPage())
                forums.add(forum);

            for (DiscussionForum forum : forums) {
                if (forum.getTitle().equals(title)) {
                    ourForum = forum;
                    break;
                }
            }

            if (ourForum == null) {
                if (forumtry > 0) {
                    System.out.println("oops, forum still not there the second time");
                    return null;
                }
                forumtry++;

                // if a new site, may need to create the area or we'll get a backtrace when creating forum
                areaManager.getDiscussionArea(siteId);

                ourForum = discussionForumManager.createForum();
                ourForum.setTitle(title);
                discussionForumManager.saveForum(siteId, ourForum);

                continue; // reread, better be there this time

            }

            // forum now exists, and was just reread

            ourTopic = null;

            for (Object o : ourForum.getTopicsSet()) {
                DiscussionTopic topic = (DiscussionTopic) o;
                if (topic.getTitle().equals(topicTitle)) {
                    ourTopic = topic;
                    break;
                }
            }

            if (ourTopic != null) // ok, forum and topic exist
                break;

            if (topictry > 0) {
                System.out.println("oops, topic still not there the second time");
                return null;
            }
            topictry++;

            // create it

            ourTopic = discussionForumManager.createTopic(ourForum);
            ourTopic.setTitle(topicTitle);

            if (attachmentHrefs != null && attachmentHrefs.size() > 0) {
                for (String href : attachmentHrefs) {
                    // we don't have any real label for attachments. About all we can do is use the filename, without path
                    String label = href;
                    int slash = label.lastIndexOf("/");
                    if (slash >= 0)
                        label = label.substring(slash + 1);
                    if (label.equals(""))
                        label = "Attachment";

                    // basedir is a folder in contenthosting starting with /group/ where our content has been loaded
                    // discussionForum needs a Sakai content resource ID for the attachment
                    Attachment thisDFAttach = discussionForumManager
                            .createDFAttachment(removeDotDot(baseDir + href), label);
                    ourTopic.addAttachment(thisDFAttach);
                }
            }

            String shortText = null;
            if (texthtml) {
                ourTopic.setExtendedDescription(text.replaceAll("\\$IMS-CC-FILEBASE\\$", base));
                shortText = FormattedText.convertFormattedTextToPlaintext(text);
            } else {
                ourTopic.setExtendedDescription(FormattedText.convertPlaintextToFormattedText(text));
                shortText = text;
            }
            shortText = org.apache.commons.lang.StringUtils.abbreviate(shortText, 254);

            ourTopic.setShortDescription(shortText);

            // there's a better way to do attachments, but it's too complex for now

            if (hide)
                discussionForumManager.saveTopicAsDraft(ourTopic);
            else
                discussionForumManager.saveTopic(ourTopic);

            // now go back and mmake sure everything is there

        }

        return "/" + FORUM_TOPIC + "/" + ourTopic.getId();
    }

    public boolean objectExists() {
        if (type == TYPE_FORUM_TOPIC) {
            if (topic == null)
                topic = getTopicById(true, id);
            return topic != null;
        } else {
            if (forum == null)
                forum = getForumById(true, id);
            return forum != null;
        }

    }

    public boolean notPublished(String ref) {
        return false;
    }

    public boolean notPublished() {
        if (!objectExists())
            return true;
        if (type == TYPE_FORUM_TOPIC) {
            return topic.getOpenForum().getDraft();
        } else {
            return ((DiscussionForum) forum).getDraft();
        }

    }

    // return the list of groups if the item is only accessible to specific groups
    // null if it's accessible to the whole site.
    public List<String> getGroups(boolean nocache) {

        // don't need cache, since simplepagebean is now caching groups
        //   List<String>ret = (List<String>)topicCache.get(id);
        //   if (!nocache && ret != null) {
        //       if (ret.size() == 0)
        //      return null;
        //       else 
        //      return ret;
        //   } else {
        //   }

        if (type != TYPE_FORUM_TOPIC)
            return null;

        List<String> ret = new ArrayList<String>();

        if (topic == null)
            topic = getTopicById(true, id);
        if (topic == null)
            return null;

        Set<DBMembershipItem> oldMembershipItemSet = uiPermissionsManager.getTopicItemsSet((DiscussionTopic) topic);

        Collection<Group> groups = null;

        try {
            Site site = SiteService.getSite(ToolManager.getCurrentPlacement().getContext());
            groups = site.getGroups();
        } catch (Exception e) {
            System.out.println("Unable to get site info for getGroups " + e);
        }

        // now change any existing ones into null
        for (DBMembershipItem item : oldMembershipItemSet) {
            if (item.getPermissionLevelName().equals("Contributor")
                    && item.getType().equals(MembershipItem.TYPE_GROUP)) {
                String name = item.getName(); // oddly, this is the actual name, not the ID
                for (Group group : groups) {
                    if (name.equals(group.getTitle()))
                        ret.add(group.getId());
                }
            }
        }

        //   topicCache.put(id, ret, DEFAULT_EXPIRATION);
        if (ret.size() == 0)
            return null;
        else
            return ret;
    }

    // set the item to be accessible only to the specific groups.
    // null to make it accessible to the whole site
    public void setGroups(Collection<String> groups) {

        if (type != TYPE_FORUM_TOPIC)
            return;

        // Setgroups with a non-null list: we set all contributor entries to none, and then set the
        //    specified groups to contribtor. By only handling groups, we avoid interfering with
        //    anything you might do in the tool. But the moment you use access control, we take
        //    over. Sorry. Once we've done that you could go back into the tool and hack, but I
        //    don't recommend that.
        // Setgroups with a null list: we set all contributor entries to none, and then set all roles
        //    other than maintain to contributor.

        setMasks();

        //System.out.println("topic 1 " + topic + " " + groups);
        //   if (topic == null)
        topic = getTopicById(true, id);
        //System.out.println("topic 2 " + topic);
        if (topic == null)
            return;

        Site site = null;
        try {
            site = SiteService.getSite(ToolManager.getCurrentPlacement().getContext());
        } catch (Exception e) {
            System.out.println("Unable to get site info for AddEntityControl " + e);
            return;
        }

        // topicCache.remove(id);

        // old entries
        Set<DBMembershipItem> oldMembershipItemSet = uiPermissionsManager.getTopicItemsSet((DiscussionTopic) topic);

        DBMembershipItem membershipItem = null;

        boolean haveOwner = false;
        boolean changed = false;

        if (groups != null && groups.size() > 0) {

            // this is the groups we've been asked to use
            // remove groups form this as we see them if they already have access
            // so at the end we just add the ones remaining
            List<String> groupNames = new ArrayList<String>();
            Set<String> addGroupNames = new HashSet<String>();
            for (String groupId : groups) {
                groupNames.add(site.getGroup(groupId).getTitle());
                addGroupNames.add(site.getGroup(groupId).getTitle());
            }
            //       System.out.println("groups " + groups + " " + groupNames + " " + addGroupNames);
            //       System.out.println("oldMembership " + oldMembershipItemSet.size());

            // delete groups from here as they are done.

            // if we've seen an owner. Otherwise set the maintain role as owner

            // Setgroups with a non-null list: we set all contributor entries to none, and then set the
            //    specified groups to contribtor. However we don't touch owner.
            // By only handling groups, we avoid interfering with
            //    anything you might do in the tool. But the moment you use access control, we take
            //    over. Sorry. Once we've done that you could go back into the tool and hack, but I
            //    don't recommend that.

            for (DBMembershipItem item : oldMembershipItemSet) {
                // kill everything except our own groups
                // this will leave the owner but remove all other roles
                //System.out.println("item " + item.getType() + " " + item.getName() + " " + item.getPermissionLevelName());
                if (item.getType().equals(MembershipItem.TYPE_GROUP) && groupNames.contains(item.getName())) {
                    //          System.out.println("found group " + item.getName());
                    addGroupNames.remove(item.getName()); // we've seen it
                    // if it's one of our groups make it a contributor if it's not already an owner
                    if (!item.getPermissionLevelName().equals("Contributor")
                            && !item.getPermissionLevelName().equals("Owner")) {

                        PermissionLevel contributorLevel = permissionLevelManager
                                .createPermissionLevel("Contributor", IdManager.createUuid(), contributorMask);
                        permissionLevelManager.savePermissionLevel(contributorLevel);

                        item.setPermissionLevel(contributorLevel);
                        item.setPermissionLevelName("Contributor");
                        permissionLevelManager.saveDBMembershipItem(item);
                    }
                } else if (!item.getPermissionLevelName().equals("Owner")) { // only group members are contributors
                    // remove contributor from anything else, both groups and roles
                    //System.out.println("set none");
                    //          System.out.println("setgroups make none " + item.getName());
                    PermissionLevel noneLevel = permissionLevelManager.createPermissionLevel("None",
                            IdManager.createUuid(), noneMask);
                    permissionLevelManager.savePermissionLevel(noneLevel);

                    item.setPermissionLevel(noneLevel);
                    item.setPermissionLevelName("None");
                    permissionLevelManager.saveDBMembershipItem(item);
                }
            }
            for (String newGroupName : addGroupNames) {
                //System.out.println("addgroup " + newGroupName);
                changed = true;
                PermissionLevel contributorLevel = permissionLevelManager.createPermissionLevel("Contributor",
                        IdManager.createUuid(), contributorMask);
                permissionLevelManager.savePermissionLevel(contributorLevel);
                membershipItem = permissionLevelManager.createDBMembershipItem(newGroupName, "Contributor",
                        MembershipItem.TYPE_GROUP);
                membershipItem.setPermissionLevel(contributorLevel);
                permissionLevelManager.saveDBMembershipItem(membershipItem);
                oldMembershipItemSet.add(membershipItem);
            }

        } else {
            // Setgroups with a null list: we set all contributor entries to none, and then set all roles
            //    to contributor.  However we don't touch Owners.

            for (DBMembershipItem item : oldMembershipItemSet) {
                if (item.getPermissionLevelName().equals("Owner")) {
                    haveOwner = true;
                } else if (item.getType().equals(MembershipItem.TYPE_ROLE)) {
                    // default state has all roles except owner as contributor
                    if (!item.getPermissionLevelName().equals("Contributor")) {
                        PermissionLevel contributorLevel = permissionLevelManager
                                .createPermissionLevel("Contributor", IdManager.createUuid(), contributorMask);
                        permissionLevelManager.savePermissionLevel(contributorLevel);

                        item.setPermissionLevel(contributorLevel);
                        item.setPermissionLevelName("Contributor");
                        permissionLevelManager.saveDBMembershipItem(item);
                    }
                } else if (!item.getPermissionLevelName().equals("None")) {
                    // kill other contributors
                    PermissionLevel noneLevel = permissionLevelManager.createPermissionLevel("None",
                            IdManager.createUuid(), noneMask);
                    permissionLevelManager.savePermissionLevel(noneLevel);

                    item.setPermissionLevel(noneLevel);
                    item.setPermissionLevelName("None");
                    permissionLevelManager.saveDBMembershipItem(item);
                }
            }
        }

        if (changed) {
            //System.out.println("changed");
            // have to refresh the topic or the save won't work
            topic = getTopicById(true, id);
            topic.setMembershipItemSet(oldMembershipItemSet);
            forumManager.saveDiscussionForumTopic((DiscussionTopic) topic);

            //       topic.setVersion(null);
            //       try {
            //      System.out.println("simplepagetool dao " + simplePageToolDao);
            //      hibernateTemplate.merge(topic);
            //       } catch (Exception e){
            //      System.out.println("Unable to save forum topic " + e);
            //       }

        }

    }

    // only used for topics
    public String getObjectId() {
        String title = getTitle();
        // fetches topic as well
        if (title == null)
            return null;

        if (type == TYPE_FORUM_TOPIC) {
            BaseForum forum = topic.getBaseForum();
            return "forum_topic/" + id + "/" + title + "\n" + forum.getTitle();
        } else {
            return "forum_forum/" + id + "/" + title;
        }
    }

    public String findObject(String objectid, Map<String, String> objectMap, String siteid) {
        if (!objectid.startsWith("forum_topic/") && !objectid.startsWith("forum_forum/")) {
            if (nextEntity != null) {
                return nextEntity.findObject(objectid, objectMap, siteid);
            }
            return null;
        }

        // isolate forum_topic/NNN from title
        int i = 0;

        if (objectid.startsWith("forum_topic/"))
            i = objectid.indexOf("/", "forum_topic/".length());
        else
            i = objectid.indexOf("/", "forum_forum/".length());
        if (i <= 0)
            return null;
        String realobjectid = objectid.substring(0, i);
        // msgcenter uses forum/ not forum_forum/
        if (objectid.startsWith("forum_forum/"))
            realobjectid = realobjectid.substring("forum_".length());

        // now see if it's in the map
        String newtopic = objectMap.get(realobjectid);
        if (newtopic != null) {
            if (objectid.startsWith("forum_forum/"))
                return "/forum_" + newtopic; // forum/ID >> /forum_forum/ID
            else
                return "/" + newtopic; // sakaiid is /forum_topic/ID
        }

        // this must be 2.8. Can't find the topic in the map
        // i is start of title
        String title = null;
        String forumtitle = null;
        if (objectid.startsWith("forum_topic/")) {
            int j = objectid.indexOf("\n");
            title = objectid.substring(i + 1, j);
            forumtitle = objectid.substring(j + 1);
        } else {
            forumtitle = objectid.substring(i + 1);
        }

        // unfortunately we have to search the topic tree to find it.
        SortedSet<DiscussionForum> forums = new TreeSet<DiscussionForum>(
                new ForumBySortIndexAscAndCreatedDateDesc());
        for (DiscussionForum forum : forumManager.getForumsForMainPage())
            forums.add(forum);

        // security. assume this is only used in places where it's OK, so skip security checks
        // ignore draft status. We want to show drafts.
        for (DiscussionForum forum : forums) {
            if (forum.getTitle().equals(forumtitle)) {
                if (title == null) // object is forum not topic
                    return "/forum_forum/" + forum.getId();
                for (Object o : forum.getTopicsSet()) {
                    DiscussionTopic topic = (DiscussionTopic) o;
                    if (topic.getTitle().equals(title)) {
                        return "/forum_topic/" + topic.getId();
                    }
                }
            }
        }

        return null;

    }

    public String getSiteId() {
        // should be this:
        // return topic.getBaseForum().getArea().getContextId();
        // but requires a hibernate session, which doesn't exit.
        String sql = "select c.context_id from MFR_TOPIC_T a,MFR_OPEN_FORUM_T b,MFR_AREA_T c where a.id=? and a.of_surrogateKey=b.id and b.surrogatekey=c.id";

        Object fields[] = new Object[1];
        fields[0] = id;

        List<String> siteIds = SqlService.dbRead(sql, fields, null);

        if (siteIds != null && siteIds.size() > 0)
            return siteIds.get(0);

        return null;

    }

}