CCM NG/ccm-cms: FolderManipulator Copy/Move (not finished yet).

git-svn-id: https://svn.libreccm.org/ccm/ccm_ng@4611 8810af33-2d31-482b-a856-94f89814c4df

Former-commit-id: 76c878866c
pull/2/head
jensp 2017-03-01 19:48:33 +00:00
parent f869e80bd1
commit 0b44ac66ce
9 changed files with 241 additions and 81 deletions

View File

@ -71,6 +71,25 @@
<scope>provided</scope>
</dependency>
<!-- Dependencies for log4j 2 -->
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
</dependency>
<dependency>
<groupId>org.apache.shiro</groupId>
<artifactId>shiro-core</artifactId>
</dependency>
<dependency>
<groupId>org.apache.shiro</groupId>
<artifactId>shiro-web</artifactId>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>

View File

@ -43,7 +43,6 @@ import java.util.Collections;
import java.util.List;
import java.util.Locale;
import java.util.Objects;
import java.util.Optional;
import java.util.stream.Collectors;
@ -87,6 +86,12 @@ public class FolderBrowserController {
@Inject
private GlobalizationHelper globalizationHelper;
@Inject
private ContentItemRepository itemRepo;
@Inject
private ContentItemManager itemManager;
@Inject
private ContentItemL10NManager itemL10NManager;
@ -96,12 +101,6 @@ public class FolderBrowserController {
@Inject
private FolderRepository folderRepo;
@Inject
private ContentItemRepository itemRepo;
@Inject
private ContentItemManager itemManager;
private Locale defaultLocale;
/**
@ -570,4 +569,41 @@ public class FolderBrowserController {
return query.getResultList();
}
/**
* Check if a folder or its subfolders contains at least one live item.
*
* @param folder The folder to check for live items.
*
* @return {@code true} if the {@code folder} or on of its subfolders
* contains at least one live item, {@code false} otherwise.
*/
@Transactional(Transactional.TxType.REQUIRED)
public boolean hasLiveItems(final Folder folder) {
Objects.requireNonNull(folder,
"Folder to check for live items can't be null.");
//Ensure that we use an non detached entity.
final Folder theFolder = folderRepo.findById(folder.getObjectId())
.orElseThrow(() -> new IllegalArgumentException(String.format(
"No folder with id %s in the database. Where did that ID come from?",
folder.getObjectId())));
final boolean hasLiveItem = theFolder.getObjects()
.stream()
.map(categorization -> categorization.getCategorizedObject())
.filter(object -> object instanceof ContentItem)
.map(object -> (ContentItem) object)
.filter(item -> itemManager.isLive(item))
.anyMatch(item -> itemManager.isLive(item));
if (hasLiveItem) {
return true;
} else {
return theFolder.getSubFolders()
.stream()
.anyMatch(currentFolder -> hasLiveItems(currentFolder));
}
}
}

View File

@ -30,7 +30,6 @@ import com.arsdigita.bebop.GridPanel;
import com.arsdigita.bebop.Label;
import com.arsdigita.bebop.Page;
import com.arsdigita.bebop.PageState;
import com.arsdigita.bebop.PaginationModelBuilder;
import com.arsdigita.bebop.Paginator;
import com.arsdigita.bebop.RequestLocal;
import com.arsdigita.bebop.Resettable;
@ -56,46 +55,36 @@ import com.arsdigita.bebop.form.SingleSelect;
import com.arsdigita.bebop.form.Submit;
import com.arsdigita.bebop.form.TextField;
import com.arsdigita.bebop.parameters.ArrayParameter;
import com.arsdigita.bebop.parameters.BigDecimalParameter;
import com.arsdigita.bebop.parameters.LongParameter;
import com.arsdigita.bebop.parameters.StringParameter;
import com.arsdigita.bebop.table.TableCellRenderer;
import com.arsdigita.bebop.table.TableColumn;
import com.arsdigita.bebop.tree.TreeCellRenderer;
import com.arsdigita.cms.CMS;
import com.arsdigita.globalization.GlobalizedMessage;
import com.arsdigita.toolbox.ui.ActionGroup;
import com.arsdigita.util.Assert;
import com.arsdigita.util.UncheckedWrapperException;
import com.arsdigita.web.Web;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.io.Writer;
import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import javax.persistence.TypedQuery;
import org.arsdigita.cms.CMSConfig;
import org.libreccm.categorization.Category;
import org.libreccm.categorization.CategoryManager;
import org.libreccm.cdi.utils.CdiUtil;
import org.libreccm.core.CcmObject;
import org.libreccm.security.PermissionChecker;
import org.libreccm.security.Shiro;
import org.libreccm.security.User;
import org.librecms.CmsConstants;
import org.librecms.contentsection.ContentItem;
import org.librecms.contentsection.ContentItemManager;
import org.librecms.contentsection.ContentItemRepository;
import org.librecms.contentsection.ContentSectionConfig;
import org.librecms.contentsection.Folder;
import org.librecms.contentsection.FolderRepository;
import org.librecms.contentsection.privileges.ItemPrivileges;
import java.util.Objects;
/**
* Browse folders and manipulate them with various actions (move/copy/delete).
*
@ -123,7 +112,7 @@ public class FolderManipulator extends SimpleContainer implements
//private static final String UNPUBLISH = "UnPublish";
private final ArrayParameter sourcesParam = new ArrayParameter(
new BigDecimalParameter(SOURCES_PARAM));
new StringParameter(SOURCES_PARAM));
private final StringParameter actionParam
= new StringParameter(ACTION_PARAM);
;
@ -174,13 +163,13 @@ public class FolderManipulator extends SimpleContainer implements
}
public final Long[] getSources(final PageState state) {
public final String[] getSources(final PageState state) {
final Long[] result = (Long[]) state.getValue(sourcesParam);
final String[] result = (String[]) state.getValue(sourcesParam);
//Return empty array instead of null.
if (result == null) {
return new Long[0];
return new String[0];
} else {
return result;
}
@ -214,9 +203,9 @@ public class FolderManipulator extends SimpleContainer implements
}
protected void moveItems(final Category target,
final Long[] itemIds) {
final String[] itemIds) {
for (Long itemId : itemIds) {
for (String itemId : itemIds) {
changeItemParent(itemId, target);
@ -224,7 +213,8 @@ public class FolderManipulator extends SimpleContainer implements
}
private void changeItemParent(final Long itemId, final Category newParent) {
private void changeItemParent(final String itemId,
final Category newParent) {
//ToDo
throw new UnsupportedOperationException();
@ -244,7 +234,7 @@ public class FolderManipulator extends SimpleContainer implements
}
protected void copyItems(final Category target,
final Long[] itemIds) {
final String[] itemIds) {
//ToDo
throw new UnsupportedOperationException();
@ -432,8 +422,9 @@ public class FolderManipulator extends SimpleContainer implements
}
@Override
public void process(final FormSectionEvent event) throws
FormProcessException {
public void process(final FormSectionEvent event)
throws FormProcessException {
final PageState state = event.getPageState();
itemView.setVisible(state, false);
@ -459,7 +450,7 @@ public class FolderManipulator extends SimpleContainer implements
targetSelector.setVisible(state, false);
final Category folder = targetSelector.getTarget(state);
final Long[] itemIds = getSources(state);
final String[] itemIds = getSources(state);
if (isCopy(state)) {
copyItems(folder, itemIds);
@ -537,7 +528,7 @@ public class FolderManipulator extends SimpleContainer implements
throw new IllegalStateException("No source items specified");
}
final Category target = targetSelector.getTarget(state);
final Folder target = targetSelector.getTarget(state);
final FormData data = event.getFormData();
if (target == null) {
data.addError(new GlobalizedMessage(
@ -564,7 +555,7 @@ public class FolderManipulator extends SimpleContainer implements
CmsConstants.CMS_FOLDER_BUNDLE);
}
for (Long source : getSources(state)) {
for (String source : getSources(state)) {
validateItem(source, target, state, data);
@ -572,37 +563,78 @@ public class FolderManipulator extends SimpleContainer implements
}
private void validateItem(final Long itemId,
final Category target,
private void validateItem(final String objectId,
final Folder target,
final PageState state,
final FormData data) {
Objects.requireNonNull(objectId, "objectId can't be null.");
final CdiUtil cdiUtil = CdiUtil.createCdiUtil();
final FolderRepository folderRepo = cdiUtil.findBean(
FolderRepository.class);
final ContentItemRepository itemRepo = cdiUtil.findBean(
ContentItemRepository.class);
final ContentItemManager itemManager = cdiUtil.findBean(
ContentItemManager.class);
final FolderBrowserController controller = cdiUtil.findBean(
FolderBrowserController.class);
final PermissionChecker permissionChecker = cdiUtil.findBean(
PermissionChecker.class);
final ContentItem item = itemRepo.findById(itemId).get();
final String name = item.getDisplayName();
final CcmObject object;
final String name;
if (objectId.startsWith("folder--")) {
final long folderId = Long.parseLong(objectId.substring(
"folder--".length()));
final Folder folder = folderRepo.findById(folderId).orElseThrow(
() -> new IllegalArgumentException(String.format(
"No folder with id %d in database.", folderId)));
final long count = itemRepo.countByNameInFolder(target, name);
if (count > 0) {
// there is an item in the target folder that already has this name
addErrorMessage(data, "cms.ui.folder.item_already_exists", name);
}
name = folder.getName();
if (itemManager.isLive(item) && isMove(state)) {
//Check if folder or subfolder contains live items
if (isMove(state) && controller.hasLiveItems(folder)) {
addErrorMessage(data, "cms.ui.folder.item_is_live", name);
}
if (!(permissionChecker.isPermitted(
ItemPrivileges.DELETE, item))
&& isMove(state)) {
addErrorMessage(data, "cms.ui.folder.no_permission_for_item",
object = folder;
} else if (objectId.startsWith("item--")) {
final long itemId = Long.parseLong(objectId.substring(
"item--".length()));
final ContentItem item = itemRepo.findById(itemId).orElseThrow(
() -> new IllegalArgumentException(String.format(
"No content item with id %d in database.", itemId)));
name = item.getDisplayName();
if (isMove(state) && itemManager.isLive(item)) {
addErrorMessage(data, "cms.ui.folder.item_is_live", name);
}
object = item;
} else {
throw new IllegalArgumentException(String.format(
"Provided objectId '%s' does not start with 'folder--' "
+ "or 'item--'.",
objectId));
}
final long count = controller.countObjects(target, name);
if (count > 0) {
// there is an item or subfolder in the target folder that already has this name
addErrorMessage(data, "cms.ui.folder.item_already_exists",
name);
}
if (!(permissionChecker.isPermitted(
ItemPrivileges.DELETE, object))
&& isMove(state)) {
addErrorMessage(data,
"cms.ui.folder.no_permission_for_item",
object.getDisplayName());
}
}
}
@ -755,8 +787,8 @@ public class FolderManipulator extends SimpleContainer implements
null);
}
public Category getTarget(final PageState state) {
return (Category) targetModel.getSelectedObject(state);
public Folder getTarget(final PageState state) {
return (Folder) targetModel.getSelectedObject(state);
}
public boolean isCancelled(final PageState state) {
@ -1072,7 +1104,25 @@ public class FolderManipulator extends SimpleContainer implements
final Object key) {
// Get the list of invalid folders once per request.
ArrayList invalidFolders = (ArrayList) m_invalidFolders.get(state);
@SuppressWarnings("unchecked")
ArrayList<String> invalidFolders
= (ArrayList<String>) m_invalidFolders.get(state);
if (invalidFolders == null) {
// The list of invalid folders has not been set for this
// request. Setting now.
invalidFolders = new ArrayList<>();
final String[] sources = getSources(state);
for (final String source : sources) {
if (source.startsWith("folder--")) {
invalidFolders.addAll(source);
}
}
}
// if (invalidFolders == null) {
// // The list of invalid folders has not been set for this

View File

@ -76,7 +76,7 @@
<artifactId>flyway-core</artifactId>
</dependency>
<!-- Dependencies for log4j 2 including adapter for the log4j 1.2 API -->
<!-- Dependencies for log4j 2 -->
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
@ -85,10 +85,6 @@
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
</dependency>
<!--<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-1.2-api</artifactId>
</dependency>-->
<dependency>
<groupId>commons-beanutils</groupId>

View File

@ -778,10 +778,17 @@ public class CategoryManager {
*
* @return The path of the category.
*/
@Transactional(Transactional.TxType.REQUIRED)
public String getCategoryPath(final Category category) {
Objects.requireNonNull(category, "Can't get the path for category null");
final List<String> tokens = new ArrayList<>();
Category current = category;
Category current = categoryRepo.findById(category.getObjectId())
.orElseThrow(() -> new IllegalArgumentException(String.format(
"No category with ID %d in the database. Where did that ID come from?",
category.getObjectId())));
while (current.getParentCategory() != null) {
tokens.add(current.getDisplayName());
current = current.getParentCategory();
@ -794,6 +801,7 @@ public class CategoryManager {
return joiner.toString();
}
@Transactional(Transactional.TxType.REQUIRED)
public boolean hasIndexObject(final Category category) {
final TypedQuery<Boolean> query = entityManager
.createNamedQuery("Categorization.hasIndexObject", Boolean.class);
@ -813,6 +821,7 @@ public class CategoryManager {
* @return An {@link Optional} containing the index object of the provided
* category if the category has an index object.
*/
@Transactional(Transactional.TxType.REQUIRED)
public Optional<CcmObject> getIndexObject(final Category category) {
if (hasIndexObject(category)) {
final TypedQuery<CcmObject> query = entityManager.createNamedQuery(

View File

@ -57,8 +57,8 @@
<scope>provided</scope>
</dependency>
<!-- Dependencies for log4j 2 including adapter for the log4j 1.2 API -->
<!--<dependency>
<!-- Dependencies for log4j 2 -->
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
</dependency>
@ -66,10 +66,15 @@
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-1.2-api</artifactId>
</dependency>-->
<groupId>org.apache.shiro</groupId>
<artifactId>shiro-core</artifactId>
</dependency>
<dependency>
<groupId>org.apache.shiro</groupId>
<artifactId>shiro-web</artifactId>
</dependency>
<dependency>
<groupId>junit</groupId>
@ -77,17 +82,6 @@
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.hamcrest</groupId>
<artifactId>hamcrest-core</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.hamcrest</groupId>
<artifactId>hamcrest-library</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.libreccm</groupId>
<artifactId>ccm-testutils</artifactId>

View File

@ -74,6 +74,15 @@
<artifactId>log4j-api</artifactId>
</dependency>
<dependency>
<groupId>org.apache.shiro</groupId>
<artifactId>shiro-core</artifactId>
</dependency>
<dependency>
<groupId>org.apache.shiro</groupId>
<artifactId>shiro-web</artifactId>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>

View File

@ -0,0 +1,47 @@
// can be used to ensure that input values will be valid when used as part of urls
//
function urlize(title) {
var result = "";
for (var i = 0; i < title.length; i++) {
result = result + substitute(title.charAt(i));
}
return escape(result);
}
function substitute(c) {
var sourceChars="0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ-_ &/";
var targetChars="0123456789abcdefghijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyz-_---";
var indexChar = sourceChars.indexOf(c);
if (indexChar > -1) {
return targetChars.charAt(indexChar);
} else {
// Replacement list for special characters
switch(c) {
case "Ä": return "ae";
break;
case "ä": return "ae";
break;
case "Ö": return "oe";
break;
case "ö": return "oe";
break;
case "Ü": return "ue";
break;
case "ü": return "ue";
break;
case "ß": return "ss";
break;
default: return "";
}
}
}

2
ccm.sh
View File

@ -410,7 +410,7 @@ run() {
bundle="$3"
else
runtime="wildfly"
bundle="$2"
bundle="$1"
fi
if [ -z $bundle ]; then