org.apache.openaz.xacml.admin.model.GitRepositoryContainer.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.openaz.xacml.admin.model.GitRepositoryContainer.java

Source

/*
 *  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.
 *
 */

/*
 * Copyright 2000-2013 Vaadin Ltd.
 * 
 * Licensed 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.openaz.xacml.admin.model;

import java.io.File;
import java.io.FilenameFilter;
import java.io.IOException;
import java.io.Serializable;
import java.lang.reflect.Method;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Date;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.eclipse.jgit.api.Git;
import org.eclipse.jgit.api.Status;
import org.eclipse.jgit.api.errors.GitAPIException;
import org.eclipse.jgit.errors.NoWorkTreeException;

import org.apache.openaz.xacml.util.XACMLPolicyScanner;
import com.vaadin.data.Container;
import com.vaadin.data.Item;
import com.vaadin.data.Property;
import com.vaadin.data.util.MethodProperty;
import com.vaadin.server.Resource;
import com.vaadin.ui.TextArea;
import com.vaadin.util.FileTypeResolver;

/**
 * A hierarchical container wrapper for a filesystem.
 * 
 * @author Vaadin Ltd.
 * @since 3.0
 */
@SuppressWarnings("serial")
public class GitRepositoryContainer extends ItemSetChangeNotifier implements Container.Hierarchical {
    private Log logger = LogFactory.getLog(GitRepositoryContainer.class);

    /**
     * String identifier of a file's "name" property.
     */
    public static String PROPERTY_NAME = "Name";

    /**
     * String identifier of a file's "size" property.
     */
    public static String PROPERTY_SIZE = "Size";

    /**
     * String identifier of a file's "icon" property.
     */
    public static String PROPERTY_ICON = "Icon";

    /**
     * String identifier of a file's "last modified" property.
     */
    public static String PROPERTY_LASTMODIFIED = "Last Modified";

    /**
     * String identifier of a file's "version" property.
     */
    public static String PROPERTY_VERSION = "Version";

    /**
     * String identifier of a file's "status" property.
     */
    public static String PROPERTY_STATUS = "Status";

    /**
     * String identifier of a file's "data" property.
     */
    public static String PROPERTY_DATA = "Data";

    /**
     * List of the string identifiers for the available properties.
     */
    public static Collection<String> FILE_PROPERTIES;

    private final static Method FILEITEM_LASTMODIFIED;

    private final static Method FILEITEM_NAME;

    private final static Method FILEITEM_ICON;

    private final static Method FILEITEM_SIZE;

    private final static Method FILEITEM_VERSION;

    private final static Method FILEITEM_STATUS;

    private final static Method FILEITEM_DATA;

    static {

        FILE_PROPERTIES = new ArrayList<String>();
        FILE_PROPERTIES.add(PROPERTY_NAME);
        FILE_PROPERTIES.add(PROPERTY_ICON);
        FILE_PROPERTIES.add(PROPERTY_SIZE);
        FILE_PROPERTIES.add(PROPERTY_LASTMODIFIED);
        FILE_PROPERTIES.add(PROPERTY_VERSION);
        FILE_PROPERTIES.add(PROPERTY_STATUS);
        FILE_PROPERTIES.add(PROPERTY_DATA);
        FILE_PROPERTIES = Collections.unmodifiableCollection(FILE_PROPERTIES);
        try {
            FILEITEM_VERSION = FileItem.class.getMethod("getVersion", new Class[] {});
            FILEITEM_LASTMODIFIED = FileItem.class.getMethod("lastModified", new Class[] {});
            FILEITEM_NAME = FileItem.class.getMethod("getName", new Class[] {});
            FILEITEM_ICON = FileItem.class.getMethod("getIcon", new Class[] {});
            FILEITEM_SIZE = FileItem.class.getMethod("getSize", new Class[] {});
            FILEITEM_STATUS = FileItem.class.getMethod("getStatus", new Class[] {});
            FILEITEM_DATA = FileItem.class.getMethod("getData", new Class[] {});
        } catch (final NoSuchMethodException e) {
            throw new RuntimeException("Internal error finding methods in FilesystemContainer");
        }
    }

    private File[] roots = new File[] {};

    private FilenameFilter filter = null;

    private boolean recursive = true;

    private Path repository = null;

    /**
     * Constructs a new <code>FileSystemContainer</code> with the specified file
     * as the root of the filesystem. The files are included recursively.
     * 
     * @param root
     *            the root file for the new file-system container. Null values
     *            are ignored.
     */
    public GitRepositoryContainer(Path repository, File root) {
        super();
        this.repository = repository;
        if (root != null) {
            roots = new File[] { root };
        }
        this.setContainer(this);
    }

    /**
     * Constructs a new <code>FileSystemContainer</code> with the specified file
     * as the root of the filesystem. The files are included recursively.
     * 
     * @param root
     *            the root file for the new file-system container.
     * @param recursive
     *            should the container recursively contain subdirectories.
     */
    public GitRepositoryContainer(Path repository, File root, boolean recursive) {
        this(repository, root);
        setRecursive(recursive);
        this.setContainer(this);
    }

    /**
     * Constructs a new <code>FileSystemContainer</code> with the specified file
     * as the root of the filesystem.
     * 
     * @param root
     *            the root file for the new file-system container.
     * @param extension
     *            the Filename extension (w/o separator) to limit the files in
     *            container.
     * @param recursive
     *            should the container recursively contain subdirectories.
     */
    public GitRepositoryContainer(Path repository, File root, String extension, boolean recursive) {
        this(repository, root);
        this.setFilter(extension);
        setRecursive(recursive);
        this.setContainer(this);
    }

    /**
     * Constructs a new <code>FileSystemContainer</code> with the specified root
     * and recursivity status.
     * 
     * @param root
     *            the root file for the new file-system container.
     * @param filter
     *            the Filename filter to limit the files in container.
     * @param recursive
     *            should the container recursively contain subdirectories.
     */
    public GitRepositoryContainer(Path repository, File root, FilenameFilter filter, boolean recursive) {
        this(repository, root);
        this.setFilter(filter);
        setRecursive(recursive);
    }

    /**
     * Adds new root file directory. Adds a file to be included as root file
     * directory in the <code>FilesystemContainer</code>.
     * 
     * @param root
     *            the File to be added as root directory. Null values are
     *            ignored.
     */
    public void addRoot(File root) {
        if (root != null) {
            final File[] newRoots = new File[roots.length + 1];
            for (int i = 0; i < roots.length; i++) {
                newRoots[i] = roots[i];
            }
            newRoots[roots.length] = root;
            roots = newRoots;
        }
    }

    /**
     * Tests if the specified Item in the container may have children. Since a
     * <code>FileSystemContainer</code> contains files and directories, this
     * method returns <code>true</code> for directory Items only.
     * 
     * @param itemId
     *            the id of the item.
     * @return <code>true</code> if the specified Item is a directory,
     *         <code>false</code> otherwise.
     */
    @Override
    public boolean areChildrenAllowed(Object itemId) {
        if (logger.isTraceEnabled()) {
            logger.trace("areChildrenAllowed: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName());
        }
        return itemId instanceof File && ((File) itemId).canRead() && ((File) itemId).isDirectory();
    }

    /*
     * Gets the ID's of all Items who are children of the specified Item. Don't
     * add a JavaDoc comment here, we use the default documentation from
     * implemented interface.
     */
    @Override
    public Collection<File> getChildren(Object itemId) {

        if (!(itemId instanceof File)) {
            return Collections.unmodifiableCollection(new LinkedList<File>());
        }
        File[] f;
        if (filter != null) {
            f = ((File) itemId).listFiles(filter);
        } else {
            f = ((File) itemId).listFiles();
        }

        if (f == null) {
            return Collections.unmodifiableCollection(new LinkedList<File>());
        }

        final List<File> l = Arrays.asList(f);
        Collections.sort(l);

        return Collections.unmodifiableCollection(l);
    }

    /*
     * Gets the parent item of the specified Item. Don't add a JavaDoc comment
     * here, we use the default documentation from implemented interface.
     */
    @Override
    public Object getParent(Object itemId) {
        if (logger.isTraceEnabled()) {
            logger.trace("getParent: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName());
        }

        if (!(itemId instanceof File)) {
            return null;
        }
        return ((File) itemId).getParentFile();
    }

    /*
     * Tests if the specified Item has any children. Don't add a JavaDoc comment
     * here, we use the default documentation from implemented interface.
     */
    @Override
    public boolean hasChildren(Object itemId) {
        if (logger.isTraceEnabled()) {
            logger.trace("hasChildren: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName());
        }

        if (!(itemId instanceof File)) {
            return false;
        }
        String[] l;
        if (filter != null) {
            l = ((File) itemId).list(filter);
        } else {
            l = ((File) itemId).list();
        }
        return l != null && l.length > 0;
    }

    /*
     * Tests if the specified Item is the root of the filesystem. Don't add a
     * JavaDoc comment here, we use the default documentation from implemented
     * interface.
     */
    @Override
    public boolean isRoot(Object itemId) {
        if (logger.isTraceEnabled()) {
            logger.trace("isRoot: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName());
        }

        if (!(itemId instanceof File)) {
            return false;
        }
        for (int i = 0; i < roots.length; i++) {
            if (roots[i].equals(itemId)) {
                return true;
            }
        }
        return false;
    }

    /*
     * Gets the ID's of all root Items in the container. Don't add a JavaDoc
     * comment here, we use the default documentation from implemented
     * interface.
     */
    @Override
    public Collection<File> rootItemIds() {

        File[] f;

        // in single root case we use children
        if (roots.length == 1) {
            if (filter != null) {
                f = roots[0].listFiles(filter);
            } else {
                f = roots[0].listFiles();
            }
        } else {
            f = roots;
        }

        if (f == null) {
            return Collections.unmodifiableCollection(new LinkedList<File>());
        }

        final List<File> l = Arrays.asList(f);
        Collections.sort(l);

        return Collections.unmodifiableCollection(l);
    }

    /**
     * Returns <code>false</code> when conversion from files to directories is
     * not supported.
     * 
     * @param itemId
     *            the ID of the item.
     * @param areChildrenAllowed
     *            the boolean value specifying if the Item can have children or
     *            not.
     * @return <code>true</code> if the operaton is successful otherwise
     *         <code>false</code>.
     * @throws UnsupportedOperationException
     *             if the setChildrenAllowed is not supported.
     */
    @Override
    public boolean setChildrenAllowed(Object itemId, boolean areChildrenAllowed)
            throws UnsupportedOperationException {

        throw new UnsupportedOperationException("Conversion file to/from directory is not supported");
    }

    /**
     * Returns <code>false</code> when moving files around in the filesystem is
     * not supported.
     * 
     * @param itemId
     *            the ID of the item.
     * @param newParentId
     *            the ID of the Item that's to be the new parent of the Item
     *            identified with itemId.
     * @return <code>true</code> if the operation is successful otherwise
     *         <code>false</code>.
     * @throws UnsupportedOperationException
     *             if the setParent is not supported.
     */
    @Override
    public boolean setParent(Object itemId, Object newParentId) throws UnsupportedOperationException {
        if (logger.isTraceEnabled()) {
            logger.trace("setParent: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName() + " to: "
                    + ((File) newParentId).hashCode() + " " + ((File) newParentId).getName());
        }

        Path path = Paths.get(((File) itemId).getAbsolutePath());
        Path parent = Paths.get(((File) newParentId).getAbsolutePath());
        boolean ok = path.getParent() == parent;

        if (ok) {
            fireItemSetChange();
        }
        return ok;
    }

    /*
     * Tests if the filesystem contains the specified Item. Don't add a JavaDoc
     * comment here, we use the default documentation from implemented
     * interface.
     */
    @Override
    public boolean containsId(Object itemId) {
        if (logger.isTraceEnabled()) {
            logger.trace("containsId: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName());
        }

        if (!(itemId instanceof File)) {
            return false;
        }
        boolean val = false;

        // Try to match all roots
        for (int i = 0; i < roots.length; i++) {
            try {
                val |= ((File) itemId).getCanonicalPath().startsWith(roots[i].getCanonicalPath());
            } catch (final IOException e) { //NOPMD
                // Exception ignored
            }

        }
        if (val && filter != null) {
            val &= filter.accept(((File) itemId).getParentFile(), ((File) itemId).getName());
        }
        return val;
    }

    /*
     * Gets the specified Item from the filesystem. Don't add a JavaDoc comment
     * here, we use the default documentation from implemented interface.
     */
    @Override
    public Item getItem(Object itemId) {

        if (logger.isTraceEnabled()) {
            logger.trace("getItem: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName());
        }

        if (!(itemId instanceof File)) {
            return null;
        }
        return new FileItem((File) itemId);
    }

    public Item updateItem(Object itemId) {
        if (logger.isTraceEnabled()) {
            logger.trace("updateItem: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName());
        }

        if (!(itemId instanceof File)) {
            return null;
        }

        this.fireItemSetChange();

        return new FileItem((File) itemId);
    }

    /**
     * Internal recursive method to add the files under the specified directory
     * to the collection.
     * 
     * @param col
     *            the collection where the found items are added
     * @param f
     *            the root file where to start adding files
     */
    private void addItemIds(Collection<File> col, File f) {
        File[] l;
        if (filter != null) {
            l = f.listFiles(filter);
        } else {
            l = f.listFiles();
        }
        if (l == null) {
            // File.listFiles returns null if File does not exist or if there
            // was an IO error (permission denied)
            return;
        }
        final List<File> ll = Arrays.asList(l);
        Collections.sort(ll);

        for (final Iterator<File> i = ll.iterator(); i.hasNext();) {
            final File lf = i.next();
            col.add(lf);
            if (lf.isDirectory()) {
                addItemIds(col, lf);
            }
        }
    }

    /*
     * Gets the IDs of Items in the filesystem. Don't add a JavaDoc comment
     * here, we use the default documentation from implemented interface.
     */
    @Override
    public Collection<File> getItemIds() {

        if (recursive) {
            final Collection<File> col = new ArrayList<File>();
            for (int i = 0; i < roots.length; i++) {
                addItemIds(col, roots[i]);
            }
            return Collections.unmodifiableCollection(col);
        } else {
            File[] f;
            if (roots.length == 1) {
                if (filter != null) {
                    f = roots[0].listFiles(filter);
                } else {
                    f = roots[0].listFiles();
                }
            } else {
                f = roots;
            }

            if (f == null) {
                return Collections.unmodifiableCollection(new LinkedList<File>());
            }

            final List<File> l = Arrays.asList(f);
            Collections.sort(l);
            return Collections.unmodifiableCollection(l);
        }

    }

    /**
     * Gets the specified property of the specified file Item. The available
     * file properties are "Name", "Size" and "Last Modified". If propertyId is
     * not one of those, <code>null</code> is returned.
     * 
     * @param itemId
     *            the ID of the file whose property is requested.
     * @param propertyId
     *            the property's ID.
     * @return the requested property's value, or <code>null</code>
     */
    @Override
    public Property<?> getContainerProperty(Object itemId, Object propertyId) {

        if (!(itemId instanceof File)) {
            return null;
        }

        if (propertyId.equals(PROPERTY_NAME)) {
            return new MethodProperty<Object>(getType(propertyId), new FileItem((File) itemId), FILEITEM_NAME,
                    null);
        }

        if (propertyId.equals(PROPERTY_ICON)) {
            return new MethodProperty<Object>(getType(propertyId), new FileItem((File) itemId), FILEITEM_ICON,
                    null);
        }

        if (propertyId.equals(PROPERTY_SIZE)) {
            return new MethodProperty<Object>(getType(propertyId), new FileItem((File) itemId), FILEITEM_SIZE,
                    null);
        }

        if (propertyId.equals(PROPERTY_LASTMODIFIED)) {
            return new MethodProperty<Object>(getType(propertyId), new FileItem((File) itemId),
                    FILEITEM_LASTMODIFIED, null);
        }

        if (propertyId.equals(PROPERTY_VERSION)) {
            return new MethodProperty<Object>(getType(propertyId), new FileItem((File) itemId), FILEITEM_VERSION,
                    null);
        }

        if (propertyId.equals(PROPERTY_STATUS)) {
            return new MethodProperty<Object>(getType(propertyId), new FileItem((File) itemId), FILEITEM_STATUS,
                    null);
        }

        if (propertyId.equals(PROPERTY_DATA)) {
            return new MethodProperty<Object>(getType(propertyId), new FileItem((File) itemId), FILEITEM_DATA,
                    null);
        }

        return null;
    }

    /**
     * Gets the collection of available file properties.
     * 
     * @return Unmodifiable collection containing all available file properties.
     */
    @Override
    public Collection<String> getContainerPropertyIds() {
        return FILE_PROPERTIES;
    }

    /**
     * Gets the specified property's data type. "Name" is a <code>String</code>,
     * "Size" is a <code>Long</code>, "Last Modified" is a <code>Date</code>. If
     * propertyId is not one of those, <code>null</code> is returned.
     * 
     * @param propertyId
     *            the ID of the property whose type is requested.
     * @return data type of the requested property, or <code>null</code>
     */
    @Override
    public Class<?> getType(Object propertyId) {

        if (propertyId.equals(PROPERTY_NAME)) {
            return String.class;
        }
        if (propertyId.equals(PROPERTY_ICON)) {
            return Resource.class;
        }
        if (propertyId.equals(PROPERTY_SIZE)) {
            return Long.class;
        }
        if (propertyId.equals(PROPERTY_LASTMODIFIED)) {
            return Date.class;
        }
        if (propertyId.equals(PROPERTY_VERSION)) {
            return String.class;
        }
        if (propertyId.equals(PROPERTY_STATUS)) {
            return TextArea.class;
        }
        if (propertyId.equals(PROPERTY_DATA)) {
            return Object.class;
        }
        return null;
    }

    /**
     * Internal method to recursively calculate the number of files under a root
     * directory.
     * 
     * @param f
     *            the root to start counting from.
     */
    private int getFileCounts(File f) {
        File[] l;
        if (filter != null) {
            l = f.listFiles(filter);
        } else {
            l = f.listFiles();
        }

        if (l == null) {
            return 0;
        }
        int ret = l.length;
        for (int i = 0; i < l.length; i++) {
            if (l[i].isDirectory()) {
                ret += getFileCounts(l[i]);
            }
        }
        return ret;
    }

    /**
     * Gets the number of Items in the container. In effect, this is the
     * combined amount of files and directories.
     * 
     * @return Number of Items in the container.
     */
    @Override
    public int size() {

        if (recursive) {
            int counts = 0;
            for (int i = 0; i < roots.length; i++) {
                counts += getFileCounts(roots[i]);
            }
            return counts;
        } else {
            File[] f;
            if (roots.length == 1) {
                if (filter != null) {
                    f = roots[0].listFiles(filter);
                } else {
                    f = roots[0].listFiles();
                }
            } else {
                f = roots;
            }

            if (f == null) {
                return 0;
            }
            return f.length;
        }
    }

    /**
     * A Item wrapper for files in a filesystem.
     * 
     * @author Vaadin Ltd.
     * @since 3.0
     */
    public class FileItem implements Item {

        /**
         * The wrapped file.
         */
        private final File file;

        private Object data = null;

        /**
         * Constructs a FileItem from a existing file.
         */
        private FileItem(File file) {
            if (logger.isTraceEnabled()) {
                logger.trace("FileItem constructor: " + file.hashCode() + " " + file.getName());
            }
            this.file = file;
        }

        /*
         * Gets the specified property of this file. Don't add a JavaDoc comment
         * here, we use the default documentation from implemented interface.
         */
        @Override
        public Property<?> getItemProperty(Object id) {
            return getContainerProperty(file, id);
        }

        /*
         * Gets the IDs of all properties available for this item Don't add a
         * JavaDoc comment here, we use the default documentation from
         * implemented interface.
         */
        @Override
        public Collection<String> getItemPropertyIds() {
            return getContainerPropertyIds();
        }

        /**
         * Calculates a integer hash-code for the Property that's unique inside
         * the Item containing the Property. Two different Properties inside the
         * same Item contained in the same list always have different
         * hash-codes, though Properties in different Items may have identical
         * hash-codes.
         * 
         * @return A locally unique hash-code as integer
         */
        @Override
        public int hashCode() {
            return file.hashCode() ^ GitRepositoryContainer.this.hashCode();
        }

        /**
         * Tests if the given object is the same as the this object. Two
         * Properties got from an Item with the same ID are equal.
         * 
         * @param obj
         *            an object to compare with this object.
         * @return <code>true</code> if the given object is the same as this
         *         object, <code>false</code> if not
         */
        @Override
        public boolean equals(Object obj) {
            if (obj == null || !(obj instanceof FileItem)) {
                return false;
            }
            final FileItem fi = (FileItem) obj;
            return fi.getHost() == getHost() && fi.file.equals(file);
        }

        /**
         * Gets the host of this file.
         */
        private GitRepositoryContainer getHost() {
            return GitRepositoryContainer.this;
        }

        /**
         * Gets the file's version
         * 
         * @return Integer
         */

        public String getVersion() {
            /*
             * If its a directory, there is no version
             */
            if (this.file.isDirectory()) {
                return null;
            }
            try {
                return XACMLPolicyScanner.getVersion(Paths.get(this.file.getAbsolutePath()));
            } catch (IOException e) {
                logger.error("Could not get version: " + e);
                return "n/a";
            }
        }

        /**
         * Gets the last modified date of this file.
         * 
         * @return Date
         */
        public Date lastModified() {
            return new Date(file.lastModified());
        }

        /**
         * Gets the name of this file.
         * 
         * @return file name of this file.
         */
        public String getName() {
            return file.getName();
        }

        public File getFile() {
            return file;
        }

        /**
         * Gets the icon of this file.
         * 
         * @return the icon of this file.
         */
        public Resource getIcon() {
            return FileTypeResolver.getIcon(file);
        }

        /**
         * Gets the size of this file.
         * 
         * @return size
         */
        public Long getSize() {
            if (file.isDirectory()) {
                return null;
            }
            return file.length();
        }

        /**
         * Gets the status of this file.
         * 
         * @return status of this file.
         */
        public TextArea getStatus() {
            TextArea area = null;
            try {
                Path repoPath = this.getHost().repository;
                Git git = Git.open(repoPath.toFile());

                //
                // I would like to use absolutePath, but that seems to barf when
                // we try to relativize this if a full path is not given.
                //
                Path relativePath = repoPath.relativize(Paths.get(this.file.getPath()));

                Status status = git.status().addPath(relativePath.toString()).call();
                if (logger.isDebugEnabled()) {
                    logger.debug(this.file.getAbsolutePath());
                    logger.debug("Added: " + status.getAdded());
                    logger.debug("Changed: " + status.getChanged());
                    logger.debug("Conflicting: " + status.getConflicting());
                    logger.debug("Missing: " + status.getMissing());
                    logger.debug("Modified: " + status.getModified());
                    logger.debug("Removed: " + status.getRemoved());
                    logger.debug("Uncommitted: " + status.getUncommittedChanges());
                    logger.debug("Untracked: " + status.getUntracked());
                    logger.debug("Untracked folders; " + status.getUntrackedFolders());
                }
                //
                // Are we a file or directory?
                //
                StringBuffer buffer = new StringBuffer();
                int length = 0;
                if (this.file.isFile()) {
                    if (status.getAdded().contains(relativePath.toString())) {
                        buffer.append("Added" + "\n");
                        length++;
                    }
                    if (status.getChanged().contains(relativePath.toString())) {
                        buffer.append("Changed" + "\n");
                        length++;
                    }
                    if (status.getConflicting().contains(relativePath.toString())) {
                        buffer.append("Conflicting" + "\n");
                        length++;
                    }
                    if (status.getMissing().contains(relativePath.toString())) {
                        buffer.append("Missing" + "\n");
                        length++;
                    }
                    if (status.getModified().contains(relativePath.toString())) {
                        buffer.append("Modified" + "\n");
                        length++;
                    }
                    if (status.getRemoved().contains(relativePath.toString())) {
                        buffer.append("Removed" + "\n");
                        length++;
                    }
                    if (status.getUncommittedChanges().contains(relativePath.toString())) {
                        buffer.append("Uncommitted" + "\n");
                        length++;
                    }
                    if (status.getUntracked().contains(relativePath.toString())) {
                        buffer.append("Untracked (New)" + "\n");
                        length++;
                    }
                    if (status.getUntrackedFolders().contains(relativePath.toString())) {
                        buffer.append("Untracked Folders (New)" + "\n");
                        length++;
                    }
                } else if (this.file.isDirectory()) {
                    if (status.getUntracked().size() > 0) {
                        buffer.append("Untracked (New)" + "\n");
                        length++;
                    }
                    if (status.getUntrackedFolders().size() > 0) {
                        buffer.append("Untracked Folders (New)" + "\n");
                        length++;
                    }
                }
                if (length > 0) {
                    area = new TextArea();
                    area.setValue(buffer.toString().trim());
                    area.setWidth("100.0%");
                    area.setRows(length);
                    area.setReadOnly(true);
                }
            } catch (IOException | NoWorkTreeException | GitAPIException e) {
                logger.error(e);
            }
            return area;
        }

        /**
         * Gets the file's data
         * 
         * @return file data
         */
        public Object getData() {
            return this.data;
        }

        /**
         * @see java.lang.Object#toString()
         */
        @Override
        public String toString() {
            //            if ("".equals(file.getName())) {
            //               return file.getAbsolutePath();
            //          }
            return file.getName();
        }

        /**
         * Filesystem container does not support adding new properties.
         * 
         * @see com.vaadin.data.Item#addItemProperty(Object, Property)
         */
        @Override
        public boolean addItemProperty(Object id, @SuppressWarnings("rawtypes") Property property)
                throws UnsupportedOperationException {
            throw new UnsupportedOperationException("Filesystem container does not support adding new properties");
        }

        /**
         * Filesystem container does not support removing properties.
         * 
         * @see com.vaadin.data.Item#removeItemProperty(Object)
         */
        @Override
        public boolean removeItemProperty(Object id) throws UnsupportedOperationException {
            throw new UnsupportedOperationException("Filesystem container does not support property removal");
        }
    }

    /**
     * Generic file extension filter for displaying only files having certain
     * extension.
     * 
     * @author Vaadin Ltd.
     * @since 3.0
     */
    public class FileExtensionFilter implements FilenameFilter, Serializable {

        private final String filter;

        /**
         * Constructs a new FileExtensionFilter using given extension.
         * 
         * @param fileExtension
         *            the File extension without the separator (dot).
         */
        public FileExtensionFilter(String fileExtension) {
            filter = "." + fileExtension;
        }

        /**
         * Allows only files with the extension and directories.
         * 
         * @see java.io.FilenameFilter#accept(File, String)
         */
        @Override
        public boolean accept(File dir, String name) {
            if (name.endsWith(filter)) {
                return true;
            }
            return new File(dir, name).isDirectory();
        }

    }

    /**
     * Returns the file filter used to limit the files in this container.
     * 
     * @return Used filter instance or null if no filter is assigned.
     */
    public FilenameFilter getFilter() {
        return filter;
    }

    /**
     * Sets the file filter used to limit the files in this container.
     * 
     * @param filter
     *            The filter to set. <code>null</code> disables filtering.
     */
    public void setFilter(FilenameFilter filter) {
        this.filter = filter;
    }

    /**
     * Sets the file filter used to limit the files in this container.
     * 
     * @param extension
     *            the Filename extension (w/o separator) to limit the files in
     *            container.
     */
    public void setFilter(String extension) {
        filter = new FileExtensionFilter(extension);
    }

    /**
     * Is this container recursive filesystem.
     * 
     * @return <code>true</code> if container is recursive, <code>false</code>
     *         otherwise.
     */
    public boolean isRecursive() {
        return recursive;
    }

    /**
     * Sets the container recursive property. Set this to false to limit the
     * files directly under the root file.
     * <p>
     * Note : This is meaningful only if the root really is a directory.
     * </p>
     * 
     * @param recursive
     *            the New value for recursive property.
     */
    public void setRecursive(boolean recursive) {
        this.recursive = recursive;
    }

    /*
     * (non-Javadoc)
     * 
     * @see com.vaadin.data.Container#addContainerProperty(java.lang.Object,
     * java.lang.Class, java.lang.Object)
     */
    @Override
    public boolean addContainerProperty(Object propertyId, Class<?> type, Object defaultValue)
            throws UnsupportedOperationException {
        throw new UnsupportedOperationException("File system container does not support this operation");
    }

    /*
     * (non-Javadoc)
     * 
     * @see com.vaadin.data.Container#addItem()
     */
    @Override
    public Object addItem() throws UnsupportedOperationException {
        throw new UnsupportedOperationException("Git repository container does not support this operation");
    }

    /*
     * (non-Javadoc)
     * 
     * @see com.vaadin.data.Container#addItem(java.lang.Object)
     */
    @Override
    public Item addItem(Object itemId) throws UnsupportedOperationException {
        if (!(itemId instanceof File)) {
            throw new UnsupportedOperationException(
                    "Git repository container does not support this operation for Objects that are not files.");
        }
        if (logger.isTraceEnabled()) {
            logger.trace("addItem: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName());
        }

        fireItemSetChange();

        return new FileItem((File) itemId);
    }

    /*
     * (non-Javadoc)
     * 
     * @see com.vaadin.data.Container#removeAllItems()
     */
    @Override
    public boolean removeAllItems() throws UnsupportedOperationException {
        throw new UnsupportedOperationException("File system container does not support this operation");
    }

    /*
     * (non-Javadoc)
     * 
     * @see com.vaadin.data.Container#removeItem(java.lang.Object)
     */
    @Override
    public boolean removeItem(Object itemId) throws UnsupportedOperationException {

        if (logger.isTraceEnabled()) {
            logger.trace("removeItem: " + ((File) itemId).hashCode() + " " + ((File) itemId).getName());
        }

        fireItemSetChange();
        return true;
    }

    /*
     * (non-Javadoc)
     * 
     * @see com.vaadin.data.Container#removeContainerProperty(java.lang.Object )
     */
    @Override
    public boolean removeContainerProperty(Object propertyId) throws UnsupportedOperationException {
        throw new UnsupportedOperationException("File system container does not support this operation");
    }

    public Object getRoot(int index) {
        if (index >= this.roots.length) {
            return null;
        }
        return this.roots[index];
    }
}