galaxy-commits
  Threads by month 
                
            - ----- 2025 -----
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2024 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2023 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2022 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2021 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2020 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2019 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2018 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2017 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2016 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2015 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2014 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2013 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2012 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2011 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 - April
 - March
 - February
 - January
 - ----- 2010 -----
 - December
 - November
 - October
 - September
 - August
 - July
 - June
 - May
 
December 2013
- 1 participants
 - 207 discussions
 
                        
                            
                                
                            
                            commit/galaxy-central: greg: Add a tool shed container to display successful repository and tool dependency installations and improve statistics and reporting in the tool shed's install and test runs.
                        
                        
by commits-noreply@bitbucket.org 19 Dec '13
                    by commits-noreply@bitbucket.org 19 Dec '13
19 Dec '13
                    
                        1 new commit in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/a0e80a712c37/
Changeset:   a0e80a712c37
User:        greg
Date:        2013-12-19 18:49:50
Summary:     Add a tool shed container to display successful repository and tool dependency installations and improve statistics and reporting in the tool shed's install and test runs.
Affected #:  8 files
diff -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b -r a0e80a712c37754d2d33260919469a7dccb2f1b1 lib/galaxy/model/tool_shed_install/__init__.py
--- a/lib/galaxy/model/tool_shed_install/__init__.py
+++ b/lib/galaxy/model/tool_shed_install/__init__.py
@@ -265,8 +265,7 @@
         """Return the repository's tool dependencies that are currently installed, but possibly in an error state."""
         installed_dependencies = []
         for tool_dependency in self.tool_dependencies:
-            if tool_dependency.status in [ ToolDependency.installation_status.INSTALLED,
-                                           ToolDependency.installation_status.ERROR ]:
+            if tool_dependency.status in [ ToolDependency.installation_status.INSTALLED ]:
                 installed_dependencies.append( tool_dependency )
         return installed_dependencies
 
@@ -442,6 +441,16 @@
         return dependencies_being_installed
 
     @property
+    def tool_dependencies_installed_or_in_error( self ):
+        """Return the repository's tool dependencies that are currently installed, but possibly in an error state."""
+        installed_dependencies = []
+        for tool_dependency in self.tool_dependencies:
+            if tool_dependency.status in [ ToolDependency.installation_status.INSTALLED,
+                                           ToolDependency.installation_status.ERROR ]:
+                installed_dependencies.append( tool_dependency )
+        return installed_dependencies
+
+    @property
     def tool_dependencies_missing_or_being_installed( self ):
         dependencies_missing_or_being_installed = []
         for tool_dependency in self.tool_dependencies:
diff -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b -r a0e80a712c37754d2d33260919469a7dccb2f1b1 lib/galaxy/tools/__init__.py
--- a/lib/galaxy/tools/__init__.py
+++ b/lib/galaxy/tools/__init__.py
@@ -2733,7 +2733,7 @@
     def build_dependency_shell_commands( self ):
         """Return a list of commands to be run to populate the current environment to include this tools requirements."""
         if self.tool_shed_repository:
-            installed_tool_dependencies = self.tool_shed_repository.installed_tool_dependencies
+            installed_tool_dependencies = self.tool_shed_repository.tool_dependencies_installed_or_in_error
         else:
             installed_tool_dependencies = None
         return self.app.toolbox.dependency_manager.dependency_shell_commands( self.requirements,
diff -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b -r a0e80a712c37754d2d33260919469a7dccb2f1b1 lib/galaxy/webapps/galaxy/controllers/admin_toolshed.py
--- a/lib/galaxy/webapps/galaxy/controllers/admin_toolshed.py
+++ b/lib/galaxy/webapps/galaxy/controllers/admin_toolshed.py
@@ -255,7 +255,7 @@
                     tool_shed_repository.uninstalled = True
                     # Remove all installed tool dependencies and tool dependencies stuck in the INSTALLING state, but don't touch any
                     # repository dependencies.
-                    tool_dependencies_to_uninstall = tool_shed_repository.installed_tool_dependencies
+                    tool_dependencies_to_uninstall = tool_shed_repository.tool_dependencies_installed_or_in_error
                     tool_dependencies_to_uninstall.extend( tool_shed_repository.tool_dependencies_being_installed )
                     for tool_dependency in tool_dependencies_to_uninstall:
                         uninstalled, error_message = tool_dependency_util.remove_tool_dependency( trans.app, tool_dependency )
diff -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b -r a0e80a712c37754d2d33260919469a7dccb2f1b1 lib/tool_shed/util/container_util.py
--- a/lib/tool_shed/util/container_util.py
+++ b/lib/tool_shed/util/container_util.py
@@ -21,26 +21,29 @@
         self.key = key
         self.label = label
         self.parent = parent
+        self.current_repository_installation_errors = []
+        self.current_repository_successful_installations = []
         self.description = None
         self.datatypes = []
+        self.failed_tests = []
         self.folders = []
+        self.invalid_data_managers = []
         self.invalid_repository_dependencies = []
         self.invalid_tool_dependencies = []
         self.invalid_tools = []
-        self.current_repository_installation_errors = []
-        self.repository_installation_errors = []
-        self.tool_dependency_installation_errors = []
-        self.valid_tools = []
-        self.valid_data_managers = []
-        self.invalid_data_managers = []
-        self.tool_dependencies = []
-        self.failed_tests = []
         self.missing_test_components = []
         self.not_tested = []
         self.passed_tests = []
+        self.readme_files = []
+        self.repository_dependencies = []
+        self.repository_installation_errors = []
+        self.repository_successful_installations = []
         self.test_environments = []
-        self.repository_dependencies = []
-        self.readme_files = []
+        self.tool_dependencies = []
+        self.tool_dependency_installation_errors = []
+        self.tool_dependency_successful_installations = []
+        self.valid_tools = []
+        self.valid_data_managers = []
         self.workflows = []
 
     def contains_folder( self, folder ):
@@ -230,6 +233,17 @@
         self.error_message = error_message
 
 
+class RepositorySuccessfulInstallation( object ):
+    """Repository installation object"""
+
+    def __init__( self, id=None, tool_shed=None, name=None, owner=None, changeset_revision=None ):
+        self.id = id
+        self.tool_shed = tool_shed
+        self.name = name
+        self.owner = owner
+        self.changeset_revision = changeset_revision
+
+
 class TestEnvironment( object ):
     """Tool test environment object"""
 
@@ -294,6 +308,16 @@
         self.error_message = error_message
 
 
+class ToolDependencySuccessfulInstallation( object ):
+    """Tool dependency installation object"""
+
+    def __init__( self, id=None, type=None, name=None, version=None, installation_directory=None ):
+        self.id = id
+        self.type = type
+        self.name = name
+        self.version = version
+        self.installation_directory = installation_directory
+
 class Workflow( object ):
     """Workflow object."""
 
@@ -1097,7 +1121,8 @@
             #    {'python_version': '2.7.4', 'tool_shed_mercurial_version': '2.2.3', 'system': 'Linux 3.8.0-30-generic', 
             #     'tool_shed_database_version': 21, 'architecture': 'x86_64', 'galaxy_revision': '11573:a62c54ddbe2a', 
             #     'galaxy_database_version': 117, 'time_tested': '2013-12-03 09:11:48', 'tool_shed_revision': '11556:228156daa575'}, 
-            # 'installation_errors': {'current_repository': [], 'repository_dependencies': [], 'tool_dependencies': []}
+            # 'installation_errors': {'current_repository': [], 'repository_dependencies': [], 'tool_dependencies': []},
+            # 'successful_installations': {'current_repository': [], 'repository_dependencies': [], 'tool_dependencies': []}
             # }
             test_environment_dict = tool_test_results_dict.get( 'test_environment', None )
             if test_environment_dict is None:
@@ -1335,6 +1360,82 @@
                                                                                                   version=td_version,
                                                                                                   error_message=td_error_message )
                             tool_dependencies_folder.tool_dependency_installation_errors.append( tool_dependency_installation_error )
+            successful_installation_dict = tool_test_results_dict.get( 'successful_installations', {} )
+            if len( successful_installation_dict ) > 0:
+                # 'successful_installation':
+                #    {'current_repository': [], 
+                #     'repository_dependencies': [], 
+                #     'tool_dependencies': 
+                #        [{'installation_directory': 'some path' 'type': 'package', 'name': 'MIRA', 'version': '4.0'}]
+                #    }
+                current_repository_successful_installation_dicts = successful_installation_dict.get( 'current_repository', [] )
+                repository_dependency_successful_installation_dicts = successful_installation_dict.get( 'repository_dependencies', [] )
+                tool_dependency_successful_installation_dicts = successful_installation_dict.get( 'tool_dependencies', [] )
+                if len( current_repository_successful_installation_dicts ) > 0 or \
+                    len( repository_dependency_successful_installation_dicts ) > 0 or \
+                    len( tool_dependency_successful_installation_dicts ) > 0:
+                    repository_installation_success_id = 0
+                    folder_id += 1
+                    successful_installation_base_folder = Folder( id=folder_id,
+                                                                  key='successful_installations',
+                                                                  label='Successful installations',
+                                                                  parent=containing_folder )
+                    containing_folder.folders.append( successful_installation_base_folder )
+                    # Displaying the successful installation of the current repository is not really necessary, so we'll skip it.
+                    if len( repository_dependency_successful_installation_dicts ) > 0:
+                        folder_id += 1
+                        repository_dependencies_folder = Folder( id=folder_id,
+                                                                 key='repository_dependency_successful_installations',
+                                                                 label='Repository dependencies',
+                                                                 parent=successful_installation_base_folder )
+                        successful_installation_base_folder.folders.append( repository_dependencies_folder )
+                        for repository_dependency_successful_installation_dict in repository_dependency_successful_installation_dicts:
+                            repository_installation_success_id += 1
+                            try:
+                                rd_tool_shed = str( repository_dependency_successful_installation_dict.get( 'tool_shed', '' ) )
+                                rd_name = str( repository_dependency_successful_installation_dict.get( 'name', '' ) )
+                                rd_owner = str( repository_dependency_successful_installation_dict.get( 'owner', '' ) )
+                                rd_changeset_revision = str( repository_dependency_successful_installation_dict.get( 'changeset_revision', '' ) )
+                            except Exception, e:
+                                rd_tool_shed = 'unknown'
+                                rd_name = 'unknown'
+                                rd_owner = 'unknown'
+                                rd_changeset_revision = 'unknown'
+                            repository_installation_success = \
+                                RepositoryInstallationSuccess( id=repository_installation_success_id,
+                                                               tool_shed=rd_tool_shed,
+                                                               name=rd_name,
+                                                               owner=rd_owner,
+                                                               changeset_revision=rd_changeset_revision )
+                            repository_dependencies_folder.repository_successful_installations.append( repository_installation_success )
+                    if len( tool_dependency_successful_installation_dicts ) > 0:
+                        # [{'installation_directory': 'some path' 'type': 'package', 'name': 'MIRA', 'version': '4.0'}]
+                        folder_id += 1
+                        tool_dependencies_folder = Folder( id=folder_id,
+                                                           key='tool_dependency_successful_installations',
+                                                           label='Tool dependencies',
+                                                           parent=successful_installation_base_folder )
+                        successful_installation_base_folder.folders.append( tool_dependencies_folder )
+                        tool_dependency_error_id = 0
+                        for tool_dependency_successful_installation_dict in tool_dependency_successful_installation_dicts:
+                            tool_dependency_error_id += 1
+                            try:
+                                td_type = str( tool_dependency_successful_installation_dict.get( 'type', '' ) )
+                                td_name = str( tool_dependency_successful_installation_dict.get( 'name', '' ) )
+                                td_version = str( tool_dependency_successful_installation_dict.get( 'version', '' ) )
+                                td_installation_directory = tool_dependency_successful_installation_dict.get( 'installation_directory', '' )
+                            except Exception, e:
+                                td_type = 'unknown'
+                                td_name = 'unknown'
+                                td_version = 'unknown'
+                                td_installation_directory = str( e )
+                            tool_dependency_successful_installation = \
+                                ToolDependencySuccessfulInstallation( id=tool_dependency_error_id,
+                                                                      type=td_type,
+                                                                      name=td_name,
+                                                                      version=td_version,
+                                                                      installation_directory=td_installation_directory )
+                            tool_dependencies_folder.tool_dependency_successful_installations.append( tool_dependency_successful_installation )
     else:
         tool_test_results_root_folder = None
     return folder_id, tool_test_results_root_folder
diff -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b -r a0e80a712c37754d2d33260919469a7dccb2f1b1 templates/webapps/tool_shed/repository/common.mako
--- a/templates/webapps/tool_shed/repository/common.mako
+++ b/templates/webapps/tool_shed/repository/common.mako
@@ -396,17 +396,27 @@
         %for tool_dependency_installation_error in folder.tool_dependency_installation_errors:
             ${render_tool_dependency_installation_error( tool_dependency_installation_error, pad, my_row, row_counter )}
         %endfor
-    %endif 
+    %endif
+    %if folder.tool_dependency_successful_installations:
+        %for tool_dependency_successful_installation in folder.tool_dependency_successful_installations:
+            ${render_tool_dependency_successful_installation( tool_dependency_successful_installation, pad, my_row, row_counter )}
+        %endfor
+    %endif
     %if folder.repository_installation_errors:
         %for repository_installation_error in folder.repository_installation_errors:
             ${render_repository_installation_error( repository_installation_error, pad, my_row, row_counter, is_current_repository=False )}
         %endfor
-    %endif 
+    %endif
     %if folder.current_repository_installation_errors:
         %for repository_installation_error in folder.current_repository_installation_errors:
             ${render_repository_installation_error( repository_installation_error, pad, my_row, row_counter, is_current_repository=True )}
         %endfor
-    %endif 
+    %endif
+    %if folder.repository_successful_installations:
+        %for repository_successful_installation in folder.repository_successful_installations:
+            ${render_repository_successful_installation( repository_successful_installation, pad, my_row, row_counter )}
+        %endfor
+    %endif
 </%def><%def name="render_datatype( datatype, pad, parent, row_counter, row_is_header=False )">
@@ -714,6 +724,36 @@
     %></%def>
 
+<%def name="render_tool_dependency_successful_installation( successful_installation, pad, parent, row_counter, row_is_header=False )">
+    <%
+        encoded_id = trans.security.encode_id( successful_installation.id )
+    %>
+    <tr class="datasetRow"
+        %if parent is not None:
+            parent="${parent}"
+        %endif
+        id="libraryItem-rtdsi-${encoded_id}">
+        <td style="padding-left: ${pad+20}px;">
+            <table id="test_environment">
+                <tr bgcolor="#FFFFCC">
+                    <th>Type</th><th>Name</th><th>Version</th>
+                </tr>
+                <tr>
+                    <td>${successful_installation.name | h}</td>
+                    <td>${successful_installation.type | h}</td>
+                    <td>${successful_installation.version | h}</td>
+                </tr>
+                <tr><th>Installation directory</th></tr>
+                <tr><td colspan="3">${successful_installation.installation_directory | h}</td></tr>
+            </table>
+        </td>
+    </tr>
+    <%
+        my_row = row_counter.count
+        row_counter.increment()
+    %>
+</%def>
+
 <%def name="render_repository_installation_error( installation_error, pad, parent, row_counter, row_is_header=False, is_current_repository=False )"><%
         from galaxy.util import unicodify
@@ -748,6 +788,37 @@
     %></%def>
 
+<%def name="render_repository_successful_installation( successful_installation, pad, parent, row_counter, row_is_header=False, is_current_repository=False )">
+    <%
+        encoded_id = trans.security.encode_id( successful_installation.id )
+    %>
+    <tr class="datasetRow"
+        %if parent is not None:
+            parent="${parent}"
+        %endif
+        id="libraryItem-rrsi-${encoded_id}">
+        <td style="padding-left: ${pad+20}px;">
+            <table id="test_environment">
+                %if not is_current_repository:
+                    <tr bgcolor="#FFFFCC">
+                        <th>Tool shed</th><th>Name</th><th>Owner</th><th>Changeset revision</th>
+                    </tr>
+                    <tr>
+                        <td>${successful_installation.tool_shed | h}</td>
+                        <td>${successful_installation.name | h}</td>
+                        <td>${successful_installation.owner | h}</td>
+                        <td>${successful_installation.changeset_revision | h}</td>
+                    </tr>
+                %endif
+            </table>
+        </td>
+    </tr>
+    <%
+        my_row = row_counter.count
+        row_counter.increment()
+    %>
+</%def>
+
 <%def name="render_not_tested( not_tested, pad, parent, row_counter, row_is_header=False )"><%
         encoded_id = trans.security.encode_id( not_tested.id )
diff -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b -r a0e80a712c37754d2d33260919469a7dccb2f1b1 test/install_and_test_tool_shed_repositories/base/util.py
--- a/test/install_and_test_tool_shed_repositories/base/util.py
+++ b/test/install_and_test_tool_shed_repositories/base/util.py
@@ -119,7 +119,6 @@
 # a definition for that tool shed.
 galaxy_tool_shed_url = os.environ.get( 'GALAXY_INSTALL_TEST_TOOL_SHED_URL', None )
 tool_shed_api_key = os.environ.get( 'GALAXY_INSTALL_TEST_TOOL_SHED_API_KEY', None )
-exclude_list_file = os.environ.get( 'GALAXY_INSTALL_TEST_EXCLUDE_REPOSITORIES', 'install_test_exclude.xml' )
 
 if 'GALAXY_INSTALL_TEST_SECRET' not in os.environ:
     galaxy_encode_secret = 'changethisinproductiontoo'
@@ -182,6 +181,46 @@
             return passed_tests
         return []
 
+def display_repositories_by_owner( repository_dicts ):
+    # Group summary display by repository owner.
+    repository_dicts_by_owner = {}
+    for repository_dict in repository_dicts:
+        name = str( repository_dict[ 'name' ] )
+        owner = str( repository_dict[ 'owner' ] )
+        changeset_revision = str( repository_dict[ 'changeset_revision' ] )
+        if owner in repository_dicts_by_owner:
+            repository_dicts_by_owner[ owner ].append( repository_dict )
+        else:
+            repository_dicts_by_owner[ owner ] = [ repository_dict ]
+    # Display grouped summary.
+    for owner, grouped_repository_dicts in repository_dicts_by_owner.items():
+        print "# "
+        for repository_dict in grouped_repository_dicts:
+            name = str( repository_dict[ 'name' ] )
+            owner = str( repository_dict[ 'owner' ] )
+            changeset_revision = str( repository_dict[ 'changeset_revision' ] )
+            print "# Revision %s of repository %s owned by %s" % ( changeset_revision, name, owner )
+
+def display_tool_dependencies_by_name( tool_dependency_dicts ):
+    # Group summary display by repository owner.
+    tool_dependency_dicts_by_name = {}
+    for tool_dependency_dict in tool_dependency_dicts:
+        name = str( tool_dependency_dict[ 'name' ] )
+        type = str( tool_dependency_dict[ 'type' ] )
+        version = str( tool_dependency_dict[ 'version' ] )
+        if name in tool_dependency_dicts_by_name:
+            tool_dependency_dicts_by_name[ name ].append( tool_dependency_dict )
+        else:
+            tool_dependency_dicts_by_name[ name ] = [ tool_dependency_dict ]
+    # Display grouped summary.
+    for name, grouped_tool_dependency_dicts in tool_dependency_dicts_by_name.items():
+        print "# "
+        for tool_dependency_dict in grouped_tool_dependency_dicts:
+            type = str( tool_dependency_dict[ 'type' ] )
+            name = str( tool_dependency_dict[ 'name' ] )
+            version = str( tool_dependency_dict[ 'version' ] )
+            print "# %s %s version %s" % ( type, name, version )
+
 def get_api_url( base, parts=[], params=None ):
     if 'api' in parts and parts.index( 'api' ) != 0:
         parts.pop( parts.index( 'api' ) )
@@ -474,18 +513,14 @@
 def initialize_install_and_test_statistics_dict( test_framework ):
     # Initialize a dictionary for the summary that will be printed to stdout.
     install_and_test_statistics_dict = {}
+    install_and_test_statistics_dict[ 'total_repositories_processed' ] = 0
+    install_and_test_statistics_dict[ 'successful_repository_installations' ] = []
+    install_and_test_statistics_dict[ 'successful_tool_dependency_installations' ] = []
+    install_and_test_statistics_dict[ 'repositories_with_installation_error' ] = []
+    install_and_test_statistics_dict[ 'tool_dependencies_with_installation_error' ] = []
     if test_framework == REPOSITORIES_WITH_TOOLS:
-        install_and_test_statistics_dict[ 'total_repositories_processed' ] = 0
         install_and_test_statistics_dict[ 'all_tests_passed' ] = []
         install_and_test_statistics_dict[ 'at_least_one_test_failed' ] = []
-        install_and_test_statistics_dict[ 'successful_installations' ] = []
-        install_and_test_statistics_dict[ 'repositories_with_installation_error' ] = []
-        install_and_test_statistics_dict[ 'tool_dependencies_with_installation_error' ] = []
-    elif test_framework == TOOL_DEPENDENCY_DEFINITIONS:
-        install_and_test_statistics_dict[ 'total_repositories_processed' ] = 0
-        install_and_test_statistics_dict[ 'successful_installations' ] = []
-        install_and_test_statistics_dict[ 'repositories_with_installation_error' ] = []
-        install_and_test_statistics_dict[ 'tool_dependencies_with_installation_error' ] = []
     return install_and_test_statistics_dict
 
 def initialize_tool_tests_results_dict( app, tool_test_results_dict ):
@@ -516,6 +551,9 @@
     tool_test_results_dict[ 'installation_errors' ] = dict( current_repository=[],
                                                             repository_dependencies=[],
                                                             tool_dependencies=[] )
+    tool_test_results_dict[ 'successful_installations' ] = dict( current_repository=[],
+                                                                 repository_dependencies=[],
+                                                                 tool_dependencies=[] )
     return tool_test_results_dict
 
 def install_repository( app, repository_dict ):
@@ -563,7 +601,17 @@
 
 def parse_exclude_list( xml_filename ):
     """Return a list of repositories to exclude from testing."""
-    # This method should return a list with the following structure:
+    # This method expects an xml document that looks something like this:
+    # <?xml version="1.0"?>
+    # <blacklist>
+    #    <repositories tool_shed="http://testtoolshed.g2.bx.psu.edu">
+    #        <reason>
+    #            <text>Some reason</text>
+    #            <repository name="some_name" owner="some_owner" />
+    #        </reason>
+    #    </repositories>
+    # </blacklist>
+    # A list is returned with the following structure:
     # [{ 'reason': The default reason or the reason specified in this section,
     #    'repositories': [( name, owner, changeset revision if changeset revision else None ),
     #                     ( name, owner, changeset revision if changeset revision else None )]}]
@@ -640,26 +688,6 @@
     result = test_runner.run( tests )
     return result, test_config.plugins._plugins
 
-def show_summary_output( repository_dicts ):
-    # Group summary display by repository owner.
-    repository_dicts_by_owner = {}
-    for repository_dict in repository_dicts:
-        name = str( repository_dict[ 'name' ] )
-        owner = str( repository_dict[ 'owner' ] )
-        changeset_revision = str( repository_dict[ 'changeset_revision' ] )
-        if owner in repository_dicts_by_owner:
-            repository_dicts_by_owner[ owner ].append( repository_dict )
-        else:
-            repository_dicts_by_owner[ owner ] = [ repository_dict ]
-    # Display grouped summary.
-    for owner, grouped_repository_dicts in repository_dicts_by_owner.items():
-        print "# "
-        for repository_dict in grouped_repository_dicts:
-            name = str( repository_dict[ 'name' ] )
-            owner = str( repository_dict[ 'owner' ] )
-            changeset_revision = str( repository_dict[ 'changeset_revision' ] )
-            print "# Revision %s of repository %s owned by %s" % ( changeset_revision, name, owner )
-
 def uninstall_repository_and_repository_dependencies( app, repository_dict ):
     """Uninstall a repository and all of its repository dependencies."""
     # This method assumes that the repositor defined by the received repository_dict is not a repository
@@ -740,7 +768,7 @@
     tool_dependency_install_path = tool_dependency.installation_directory( app )
     uninstalled, error_message = tool_dependency_util.remove_tool_dependency( app, tool_dependency )
     if error_message:
-        log.debug( 'There was an error attempting to remove directory: %s' % str( tool_dependency_install_path ) )
+        log.debug( 'Error attempting to remove directory: %s' % str( tool_dependency_install_path ) )
         log.debug( error_message )
     else:
         log.debug( 'Successfully removed tool dependency installation directory: %s' % str( tool_dependency_install_path ) )
diff -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b -r a0e80a712c37754d2d33260919469a7dccb2f1b1 test/install_and_test_tool_shed_repositories/repositories_with_tools/functional_tests.py
--- a/test/install_and_test_tool_shed_repositories/repositories_with_tools/functional_tests.py
+++ b/test/install_and_test_tool_shed_repositories/repositories_with_tools/functional_tests.py
@@ -66,6 +66,9 @@
 default_galaxy_locales = 'en'
 default_galaxy_test_file_dir = "test-data"
 os.environ[ 'GALAXY_INSTALL_TEST_TMP_DIR' ] = galaxy_test_tmp_dir
+# This file is copied to the Galaxy root directory by buildbot. 
+# It is managed by cfengine and is not locally available.
+exclude_list_file = os.environ.get( 'GALAXY_INSTALL_TEST_EXCLUDE_REPOSITORIES', 'repositories_with_tools_exclude.xml' )
 
 # This script can be run in such a way that no Tool Shed database records should be changed.
 if '-info_only' in sys.argv or 'GALAXY_INSTALL_TEST_INFO_ONLY' in os.environ:
@@ -144,7 +147,7 @@
     if error_message:
         return None, error_message
     # Handle repositories not to be tested.
-    if os.path.exists( install_and_test_base_util.exclude_list_file ):
+    if os.path.exists( exclude_list_file ):
         # Entries in the exclude_list look something like this.
         # { 'reason': The default reason or the reason specified in this section,
         #   'repositories':
@@ -157,8 +160,8 @@
         # undeleted, this script will then test them the next time it runs. We don't need to check if a repository has been deleted
         # here because our call to the Tool Shed API filters by downloadable='true', in which case deleted will always be False.
         log.debug( 'Loading the list of repositories excluded from testing from the file %s...' % \
-            str( install_and_test_base_util.exclude_list_file ) )
-        exclude_list = install_and_test_base_util.parse_exclude_list( install_and_test_base_util.exclude_list_file )
+            str( exclude_list_file ) )
+        exclude_list = install_and_test_base_util.parse_exclude_list( exclude_list_file )
     else:
         exclude_list = []
     # Generate a test method that will use Twill to install each repository into the embedded Galaxy application that was
@@ -299,7 +302,7 @@
                         test_toolbox.toolbox = app.toolbox
                     else:
                         # This repository and all of its dependencies were successfully installed.
-                        install_and_test_statistics_dict[ 'successful_installations' ].append( repository_identifier_dict )
+                        install_and_test_statistics_dict[ 'successful_repository_installations' ].append( repository_identifier_dict )
                         # Configure and run functional tests for this repository. This is equivalent to sh run_functional_tests.sh -installed
                         remove_install_tests()
                         log.debug( 'Installation of %s succeeded, running all defined functional tests.' % str( repository.name ) )
@@ -560,33 +563,33 @@
         total_repositories_processed = install_and_test_statistics_dict[ 'total_repositories_processed' ]
         all_tests_passed = install_and_test_statistics_dict[ 'all_tests_passed' ]
         at_least_one_test_failed = install_and_test_statistics_dict[ 'at_least_one_test_failed' ]
-        successful_installations = install_and_test_statistics_dict[ 'successful_installations' ]
+        successful_repository_installations = install_and_test_statistics_dict[ 'successful_repository_installations' ]
         repositories_with_installation_error = install_and_test_statistics_dict[ 'repositories_with_installation_error' ]
         tool_dependencies_with_installation_error = install_and_test_statistics_dict[ 'tool_dependencies_with_installation_error' ]
         now = time.strftime( "%Y-%m-%d %H:%M:%S" )
         print "####################################################################################"
         print "# %s - installation script for repositories containing tools completed." % now
         print "# Repository revisions processed: %s" % str( total_repositories_processed )
-        if successful_installations:
+        if successful_repository_installations:
             print "# ----------------------------------------------------------------------------------"
-            print "# The following %d revisions with all dependencies were successfully installed:" % len( successful_installations )
-            install_and_test_base_util.show_summary_output( successful_installations )
+            print "# The following %d revisions were successfully installed:" % len( successful_repository_installations )
+            install_and_test_base_util.display_repositories_by_owner( successful_repository_installations )
         if all_tests_passed:
             print '# ----------------------------------------------------------------------------------'
-            print "# %d repositories successfully passed all functional tests:" % len( all_tests_passed )
-            install_and_test_base_util.show_summary_output( all_tests_passed )
+            print "# The following %d revisions successfully passed all functional tests:" % len( all_tests_passed )
+            install_and_test_base_util.display_repositories_by_owner( all_tests_passed )
         if at_least_one_test_failed:
             print '# ----------------------------------------------------------------------------------'
-            print "# %d repositories failed at least 1 functional test:" % len( at_least_one_test_failed )
-            install_and_test_base_util.show_summary_output( at_least_one_test_failed )
+            print "# The following %d revisions failed at least 1 functional test:" % len( at_least_one_test_failed )
+            install_and_test_base_util.display_repositories_by_owner( at_least_one_test_failed )
         if repositories_with_installation_error:
             print '# ----------------------------------------------------------------------------------'
-            print "# %d repositories have installation errors:" % len( repositories_with_installation_error )
-            install_and_test_base_util.show_summary_output( repositories_with_installation_error )
+            print "# The following %d revisions have installation errors:" % len( repositories_with_installation_error )
+            install_and_test_base_util.display_repositories_by_owner( repositories_with_installation_error )
         if tool_dependencies_with_installation_error:
             print "# ----------------------------------------------------------------------------------"
             print "# The following %d tool dependencies have installation errors:" % len( tool_dependencies_with_installation_error )
-            install_and_test_base_util.show_summary_output( tool_dependencies_with_installation_error )
+            install_and_test_base_util.display_tool_dependencies_by_name( tool_dependencies_with_installation_error )
         print "####################################################################################"
     log.debug( "Shutting down..." )
     # Gracefully shut down the embedded web server and UniverseApplication.
diff -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b -r a0e80a712c37754d2d33260919469a7dccb2f1b1 test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
--- a/test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
+++ b/test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
@@ -64,6 +64,9 @@
 default_galaxy_locales = 'en'
 default_galaxy_test_file_dir = "test-data"
 os.environ[ 'GALAXY_INSTALL_TEST_TMP_DIR' ] = galaxy_test_tmp_dir
+# This file is copied to the Galaxy root directory by buildbot. 
+# It is managed by cfengine and is not locally available.
+exclude_list_file = os.environ.get( 'GALAXY_INSTALL_TEST_EXCLUDE_REPOSITORIES', 'tool_dependency_definition_exclude.xml' )
 
 # This script can be run in such a way that no Tool Shed database records should be changed.
 if '-info_only' in sys.argv or 'GALAXY_INSTALL_TEST_INFO_ONLY' in os.environ:
@@ -83,7 +86,7 @@
     if error_message:
         return None, error_message
     # Handle repositories not to be tested.
-    if os.path.exists( install_and_test_base_util.exclude_list_file ):
+    if os.path.exists( exclude_list_file ):
         # Entries in the exclude_list look something like this.
         # { 'reason': The default reason or the reason specified in this section,
         #   'repositories':
@@ -96,8 +99,8 @@
         # undeleted, this script will then test them the next time it runs. We don't need to check if a repository has been deleted
         # here because our call to the Tool Shed API filters by downloadable='true', in which case deleted will always be False.
         log.debug( 'Loading the list of repositories excluded from testing from the file %s...' % \
-            str( install_and_test_base_util.exclude_list_file ) )
-        exclude_list = install_and_test_base_util.parse_exclude_list( install_and_test_base_util.exclude_list_file )
+            str( exclude_list_file ) )
+        exclude_list = install_and_test_base_util.parse_exclude_list( exclude_list_file )
     else:
         exclude_list = []
     # Generate a test method that will use Twill to install each repository into the embedded Galaxy application that was
@@ -176,7 +179,8 @@
                     # Even if the repository failed to install, execute the uninstall method, in case a dependency did succeed.
                     log.debug( 'Attempting to uninstall revision %s of repository %s owned by %s.' % ( changeset_revision, name, owner ) )
                     try:
-                        repository = test_db_util.get_installed_repository_by_name_owner_changeset_revision( name, owner, changeset_revision )
+                        repository = \
+                            test_db_util.get_installed_repository_by_name_owner_changeset_revision( name, owner, changeset_revision )
                     except Exception, e:
                         error_message = 'Unable to find revision %s of repository %s owned by %s: %s.' % \
                             ( changeset_revision, name, owner, str( e ) )
@@ -199,9 +203,34 @@
                     log.debug( 'Installation failed for revision %s of repository %s owned by %s.' % \
                         ( changeset_revision, name, owner ) )
                 else:
+                    # The repository was successfully installed.
                     log.debug( 'Installation succeeded for revision %s of repository %s owned by %s.' % \
                         ( changeset_revision, name, owner ) )
+                    install_and_test_statistics_dict[ 'successful_repository_installations' ].append( repository_identifier_dict )
+                    tool_test_results_dict[ 'successful_installations' ][ 'current_repository' ]\
+                        .append( repository_identifier_dict )
+                    params = dict( test_install_error=False,
+                                   do_not_test=False )
+                    if repository.missing_repository_dependencies:
+                        params[ 'test_install_error' ] = True
+                        # Keep statistics for this repository's repository dependencies that resulted in installation errors.
+                        for missing_repository_dependency in repository.missing_repository_dependencies:
+                            tool_shed = str( missing_repository_dependency.tool_shed )
+                            name = str( missing_repository_dependency.name )
+                            owner = str( missing_repository_dependency.owner )
+                            changset_revision = str( missing_repository_dependency.changeset_revision )
+                            error_message = unicodify( missing_repository_dependency.error_message )
+                            missing_repository_dependency_info_dict = dict( tool_shed=tool_shed,
+                                                                            name=name,
+                                                                            owner=owner,
+                                                                            changset_revision=changset_revision,
+                                                                            error_message=error_message )
+                            install_and_test_statistics_dict[ 'repositories_with_installation_error' ]\
+                                .append( missing_repository_dependency_dict )
+                            tool_test_results_dict[ 'installation_errors' ][ 'repository_dependencies' ]\
+                                .append( missing_repository_dependency_info_dict )
                     if repository.missing_tool_dependencies:
+                        params[ 'test_install_error' ] = True
                         # Keep statistics for this repository's tool dependencies that resulted in installation errors.
                         for missing_tool_dependency in repository.missing_tool_dependencies:
                             name = str( missing_tool_dependency.name )
@@ -212,22 +241,47 @@
                                                                       name=name,
                                                                       version=version,
                                                                       error_message=error_message )
-                            install_and_test_statistics_dict[ 'tool_dependencies_with_installation_error' ].append( missing_tool_dependency_info_dict )
-                    else:
-                        # This repository and all of its dependencies were successfully installed.
-                        install_and_test_statistics_dict[ 'successful_installations' ].append( repository_identifier_dict )
-                        tool_test_results_dict[ 'passed_tests' ].append( repository_identifier_dict )
-                        params = dict( test_install_error=False,
-                                       do_not_test=False )
-                        # TODO: do something useful with response_dict
-                        response_dict = install_and_test_base_util.register_test_result( install_and_test_base_util.galaxy_tool_shed_url,
-                                                                                         tool_test_results_dicts,
-                                                                                         tool_test_results_dict,
-                                                                                         repository_dict,
-                                                                                         params,
-                                                                                         can_update_tool_shed )
-
-    install_and_test_statistics_dict[ 'total_repositories_processed' ] = total_repositories_processed
+                            install_and_test_statistics_dict[ 'tool_dependencies_with_installation_error' ]\
+                                .append( missing_tool_dependency_info_dict )
+                            tool_test_results_dict[ 'installation_errors' ][ 'tool_dependencies' ]\
+                                .append( missing_tool_dependency_info_dict )
+                    if repository.installed_repository_dependencies:
+                        # Keep statistics for this repository's tool dependencies that resulted in successful installations.
+                        for repository_dependency in repository.installed_repository_dependencies:
+                            tool_shed = str( repository_dependency.tool_shed )
+                            name = str( repository_dependency.name )
+                            owner = str( repository_dependency.owner )
+                            changeset_revision = str( repository_dependency.changeset_revision )
+                            repository_dependency_info_dict = dict( tool_shed=tool_shed,
+                                                                    name=name,
+                                                                    owner=owner,
+                                                                    changeset_revision=changeset_revision )
+                            install_and_test_statistics_dict[ 'successful_repository_installations' ]\
+                                .append( repository_dependency_info_dict )
+                            tool_test_results_dict[ 'successful_installations' ][ 'repository_dependencies' ]\
+                                .append( repository_dependency_info_dict )
+                    if repository.installed_tool_dependencies:
+                        # Keep statistics for this repository's tool dependencies that resulted in successful installations.
+                        for tool_dependency in repository.installed_tool_dependencies:
+                            name = str( tool_dependency.name )
+                            type = str( tool_dependency.type )
+                            version = str( tool_dependency.version )
+                            installation_directory = tool_dependency.installation_directory( app )
+                            tool_dependency_info_dict = dict( type=type,
+                                                              name=name,
+                                                              version=version,
+                                                              installation_directory=installation_directory )
+                            install_and_test_statistics_dict[ 'successful_tool_dependency_installations' ]\
+                                .append( tool_dependency_info_dict )
+                            tool_test_results_dict[ 'successful_installations' ][ 'tool_dependencies' ]\
+                                .append( tool_dependency_info_dict )
+                    # TODO: do something useful with response_dict
+                    response_dict = install_and_test_base_util.register_test_result( install_and_test_base_util.galaxy_tool_shed_url,
+                                                                                     tool_test_results_dicts,
+                                                                                     tool_test_results_dict,
+                                                                                     repository_dict,
+                                                                                     params,
+                                                                                     can_update_tool_shed )
     return install_and_test_statistics_dict, error_message
 
 def main():
@@ -446,25 +500,30 @@
         log.debug( error_message )
     else:
         total_repositories_processed = install_and_test_statistics_dict[ 'total_repositories_processed' ]
-        successful_installations = install_and_test_statistics_dict[ 'successful_installations' ]
+        successful_repository_installations = install_and_test_statistics_dict[ 'successful_repository_installations' ]
+        successful_tool_dependency_installations = install_and_test_statistics_dict[ 'successful_tool_dependency_installations' ]
         repositories_with_installation_error = install_and_test_statistics_dict[ 'repositories_with_installation_error' ]
         tool_dependencies_with_installation_error = install_and_test_statistics_dict[ 'tool_dependencies_with_installation_error' ]
         now = time.strftime( "%Y-%m-%d %H:%M:%S" )
         print "####################################################################################"
         print "# %s - installation script for repositories of type tool_dependency_definition completed." % now
         print "# Repository revisions processed: %s" % str( total_repositories_processed )
-        if successful_installations:
+        if successful_repository_installations:
             print "# ----------------------------------------------------------------------------------"
-            print "# The following %d revisions with all dependencies were successfully installed:" % len( successful_installations )
-            install_and_test_base_util.show_summary_output( successful_installations )
+            print "# The following %d revisions were successfully installed:" % len( successful_repository_installations )
+            install_and_test_base_util.display_repositories_by_owner( successful_repository_installations )
         if repositories_with_installation_error:
             print "# ----------------------------------------------------------------------------------"
             print "# The following %d revisions have installation errors:" % len( repositories_with_installation_error )
-            install_and_test_base_util.show_summary_output( repositories_with_installation_error )
+            install_and_test_base_util.successful_repository_installations( repositories_with_installation_error )
+        if successful_tool_dependency_installations:
+            print "# ----------------------------------------------------------------------------------"
+            print "# The following %d tool dependencies were successfully installed:" % len( successful_tool_dependency_installations )
+            install_and_test_base_util.display_tool_dependencies_by_name( successful_tool_dependency_installations )
         if tool_dependencies_with_installation_error:
             print "# ----------------------------------------------------------------------------------"
             print "# The following %d tool dependencies have installation errors:" % len( tool_dependencies_with_installation_error )
-            install_and_test_base_util.show_summary_output( tool_dependencies_with_installation_error )
+            install_and_test_base_util.display_tool_dependencies_by_name( tool_dependencies_with_installation_error )
         print "####################################################################################"
     log.debug( "Shutting down..." )
     # Gracefully shut down the embedded web server and UniverseApplication.
@@ -506,6 +565,36 @@
     #              "architecture": "x86_64",
     #              "system": "Darwin 12.2.0"
     #         },
+    #     "successful_installation":
+    #         {
+    #              'tool_dependencies':
+    #                  [
+    #                      {
+    #                         'type': 'Type of tool dependency, e.g. package, set_environment, etc.',
+    #                         'name': 'Name of the tool dependency.',
+    #                         'version': 'Version if this is a package, otherwise blank.',
+    #                         'installation_directory': 'The installation directory path.'
+    #                      },
+    #                  ],
+    #              'repository_dependencies':
+    #                  [
+    #                      {
+    #                         'tool_shed': 'The tool shed that this repository was installed from.',
+    #                         'name': 'The name of the repository that failed to install.',
+    #                         'owner': 'Owner of the failed repository.',
+    #                         'changeset_revision': 'Changeset revision of the failed repository.'
+    #                      },
+    #                  ],
+    #              'current_repository':
+    #                  [
+    #                      {
+    #                         'tool_shed': 'The tool shed that this repository was installed from.',
+    #                         'name': 'The name of the repository that failed to install.',
+    #                         'owner': 'Owner of the failed repository.',
+    #                         'changeset_revision': 'Changeset revision of the failed repository.'
+    #                      },
+    #                  ],
+    #        }
     #     "installation_errors":
     #         {
     #              'tool_dependencies':
@@ -537,12 +626,6 @@
     #                         'error_message': 'The error message that was returned when the repository failed to install.',
     #                      },
     #                  ],
-    #             {
-    #                 "name": "The name of the repository.",
-    #                 "owner": "The owner of the repository.",
-    #                 "changeset_revision": "The changeset revision of the repository.",
-    #                 "error_message": "The message stored in tool_dependency.error_message."
-    #             },
     #         }
     # }
     sys.exit( main() )
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0
                            
                          
                          
                            
    
                          
                        
                    
                    
                        38 new commits in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/57d618633209/
Changeset:   57d618633209
Branch:      data_library
User:        martenson
Date:        2013-10-08 22:41:08
Summary:     start data_library branch
Affected #:  3 files
diff -r b61927f340f30a636204ec560522acf9b76fd262 -r 57d618633209a35342f25a13a220fedce71f7a3f .hgignore
--- a/.hgignore
+++ b/.hgignore
@@ -115,3 +115,6 @@
 *.rej
 *~
 
+
+syntax: regexp
+^static/AAA_scratch$
\ No newline at end of file
diff -r b61927f340f30a636204ec560522acf9b76fd262 -r 57d618633209a35342f25a13a220fedce71f7a3f lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a library.
+API operations on the contents of a folder.
 """
 import logging, os, string, shutil, urllib, re, socket
 from cgi import escape, FieldStorage
@@ -11,12 +11,17 @@
 log = logging.getLogger( __name__ )
 
 class FolderContentsController( BaseAPIController, UsesLibraryMixin, UsesLibraryMixinItems ):
+    """
+    Class controls retrieval, creation and updating of folder contents.
+    """
 
     @web.expose_api
     def index( self, trans, folder_id, **kwd ):
         """
         GET /api/folders/{encoded_folder_id}/contents
+        
         Displays a collection (list) of a folder's contents (files and folders).
+        
         The /api/library_contents/{encoded_library_id}/contents
         lists everything in a library recursively, which is not what
         we want here. We could add a parameter to use the recursive
@@ -25,7 +30,11 @@
         rval = []
         current_user_roles = trans.get_current_user_roles()
 
+
         def traverse( folder ):
+            """
+            Load contents of the folder (folders and datasets).
+            """
             admin = trans.user_is_admin()
             rval = []
             for subfolder in folder.active_folders:
@@ -50,6 +59,8 @@
 
         try:
             folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( decoded_folder_id )
+#             log.debug("XXXXXXXXXXXXXXXXXXXXXXXXXXX folder.parent_library" + str(folder.parent_library.id))
+#             log.debug("XXXXXXXXXXXXXXXXXXXXXXXXXXX folder.parent_id" + str(folder.parent_id))
             parent_library = folder.parent_library
         except:
             folder = None
@@ -62,15 +73,43 @@
         if not folder or not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
             trans.response.status = 400
             return "Invalid folder id ( %s ) specified." % str( folder_id )
+        # TODO MARTEN Can it be that predecessors of current folder have different access rights? aka user shouldn't see them?
+        
+        # Search the path upwards and load the whole route of names and ids for breadcrumb purposes.
+        path_to_library = []
+        
+        def build_path ( folder ):
+#             log.debug("XXXXXXXXXXXXXXXXXXXXXXX folder.parent_id BEFORE   " + str(folder.parent_id))
+#             log.debug("XXXXXXXXXXXXXXXXXXXXXXX folder.parent_library.id BEFORE   " + str(folder.parent_library.id))
+            if ( folder.parent_id != folder.parent_library.id ) and ( folder.parent_id is not None ):
+                log.debug("XXXXXXXXXXXXXXXXXXXXXXX LOADING UPPER FOLDER WITH ID:  " + str(folder.parent_id))
+                upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_id )
+                log.debug("XXXXXXXXXXXXXXXXXXXXXXX upper_folder.id   " + str(upper_folder.id))
+                log.debug("XXXXXXXXXXXXXXXXXXXXXXX upper_folder.name   " + str(upper_folder.name))
+                path_to_library.append( ( upper_folder.id, upper_folder.name ) )
+                path_to_library.extend( build_path( upper_folder ) )
+            else:
+                pass    
+            return path_to_library
+            
+        full_path = build_path( folder )
 
         for content in traverse( folder ):
+            return_item = {}
             encoded_id = trans.security.encode_id( content.id )
             if content.api_type == 'folder':
+                encoded_parent_library_id = trans.security.encode_id( content.parent_library.id )
                 encoded_id = 'F' + encoded_id
-            rval.append( dict( id = encoded_id,
+                if content.parent_id is not None: # For folder return its parent's id for browsing back.
+                    encoded_parent_id = 'F' + trans.security.encode_id( content.parent_id )
+                    return_item.update ( dict ( parent_id = encoded_parent_id ) )
+            return_item.update( dict( id = encoded_id,
                                type = content.api_type,
                                name = content.name,
+                               library_id = encoded_parent_library_id,
+                               full_path = full_path,
                                url = url_for( 'folder_contents', folder_id=encoded_id ) ) )
+            rval.append( return_item )
         return rval
 
     @web.expose_api
diff -r b61927f340f30a636204ec560522acf9b76fd262 -r 57d618633209a35342f25a13a220fedce71f7a3f lib/galaxy/webapps/galaxy/api/libraries.py
--- a/lib/galaxy/webapps/galaxy/api/libraries.py
+++ b/lib/galaxy/webapps/galaxy/api/libraries.py
@@ -49,7 +49,7 @@
                            trans.model.Library.table.c.id.in_( accessible_restricted_library_ids ) ) )
         rval = []
         for library in query:
-            item = library.to_dict()
+            item = library.to_dict( view='element' )
             item['url'] = url_for( route, id=trans.security.encode_id( library.id ) )
             item['id'] = trans.security.encode_id( item['id'] )
             rval.append( item )
@@ -131,6 +131,9 @@
         rval['name'] = name
         rval['id'] = encoded_id
         return rval
+    
+    def edit( self, trans, payload, **kwd  ):
+        return "Not implemented yet"
 
     @web.expose_api
     def delete( self, trans, id, **kwd ):
https://bitbucket.org/galaxy/galaxy-central/commits/db8fd059cf4c/
Changeset:   db8fd059cf4c
Branch:      data_library
User:        martenson
Date:        2013-10-17 18:56:34
Summary:     working reverse traversing towards root library for every folder  = good for path building
Affected #:  1 file
diff -r 57d618633209a35342f25a13a220fedce71f7a3f -r db8fd059cf4cae9a6b7dc8bdbe7be5d071325d09 lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -27,11 +27,11 @@
         we want here. We could add a parameter to use the recursive
         style, but this is meant to act similar to an "ls" directory listing.
         """
-        rval = []
+        folder_contents = []
         current_user_roles = trans.get_current_user_roles()
 
 
-        def traverse( folder ):
+        def load_folder_contents( folder ):
             """
             Load contents of the folder (folders and datasets).
             """
@@ -76,41 +76,55 @@
         # TODO MARTEN Can it be that predecessors of current folder have different access rights? aka user shouldn't see them?
         
         # Search the path upwards and load the whole route of names and ids for breadcrumb purposes.
-        path_to_library = []
+        path_to_root = []
         
         def build_path ( folder ):
-#             log.debug("XXXXXXXXXXXXXXXXXXXXXXX folder.parent_id BEFORE   " + str(folder.parent_id))
-#             log.debug("XXXXXXXXXXXXXXXXXXXXXXX folder.parent_library.id BEFORE   " + str(folder.parent_library.id))
-            if ( folder.parent_id != folder.parent_library.id ) and ( folder.parent_id is not None ):
-                log.debug("XXXXXXXXXXXXXXXXXXXXXXX LOADING UPPER FOLDER WITH ID:  " + str(folder.parent_id))
+            path_to_root = []
+            # We are almost in root
+            log.debug( "XXXXXXXXXXXXXXXXXXXXXXX folder.parent_id: " + str( folder.parent_id ) )
+            log.debug( "XXXXXXXXXXXXXXXXXXXXXXX folder.parent_library.id: " + str( folder.parent_library.id ) )
+            if folder.parent_id is None:
+                log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ALMOST ROOT FOLDER! ADDING: " + str( folder.name ) )
+                path_to_root.append( ( folder.id, folder.name ) )
+#                 upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_library.id )
+#                 path_to_root.append( ( upper_folder.id, upper_folder.name ) )
+            else:
+            # We add the current folder and traverse up one folder.
+                log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ADDING THIS FOLDER AND TRAVERSING UP:  " + str( folder.name ) )
+                path_to_root.append( ( folder.id, folder.name ) )
                 upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_id )
-                log.debug("XXXXXXXXXXXXXXXXXXXXXXX upper_folder.id   " + str(upper_folder.id))
-                log.debug("XXXXXXXXXXXXXXXXXXXXXXX upper_folder.name   " + str(upper_folder.name))
-                path_to_library.append( ( upper_folder.id, upper_folder.name ) )
-                path_to_library.extend( build_path( upper_folder ) )
-            else:
-                pass    
-            return path_to_library
+                path_to_root.extend( build_path( upper_folder ) )
+#             path_to_root = path_to_root.reverse()
+#             return path_to_root[::-1]
+            return path_to_root
             
-        full_path = build_path( folder )
+        # Return the reversed path so it starts with the library node.
+        full_path = build_path( folder )[::-1]
 
-        for content in traverse( folder ):
+        # Go through every item in the folder and include its meta-data.
+        for content_item in load_folder_contents( folder ):
             return_item = {}
-            encoded_id = trans.security.encode_id( content.id )
-            if content.api_type == 'folder':
-                encoded_parent_library_id = trans.security.encode_id( content.parent_library.id )
+            encoded_id = trans.security.encode_id( content_item.id )
+            
+            # For folder return also hierarchy values
+            if content_item.api_type == 'folder':
+                encoded_parent_library_id = trans.security.encode_id( content_item.parent_library.id )
                 encoded_id = 'F' + encoded_id
-                if content.parent_id is not None: # For folder return its parent's id for browsing back.
-                    encoded_parent_id = 'F' + trans.security.encode_id( content.parent_id )
+                if content_item.parent_id is not None: # Return folder's parent id for browsing back.
+                    encoded_parent_id = 'F' + trans.security.encode_id( content_item.parent_id )
                     return_item.update ( dict ( parent_id = encoded_parent_id ) )
+            
+            # For every item return also the default meta-data
             return_item.update( dict( id = encoded_id,
-                               type = content.api_type,
-                               name = content.name,
+                               type = content_item.api_type,
+                               name = content_item.name,
                                library_id = encoded_parent_library_id,
                                full_path = full_path,
                                url = url_for( 'folder_contents', folder_id=encoded_id ) ) )
-            rval.append( return_item )
-        return rval
+            folder_contents.append( return_item )
+        if len( folder_contents ) == 0:
+            folder_contents.append( dict( full_path = full_path ) )
+        return folder_contents
 
     @web.expose_api
     def show( self, trans, id, library_id, **kwd ):
https://bitbucket.org/galaxy/galaxy-central/commits/1eda20e058eb/
Changeset:   1eda20e058eb
Branch:      data_library
User:        martenson
Date:        2013-10-21 18:31:38
Summary:     initial frontend commit
Affected #:  3 files
diff -r db8fd059cf4cae9a6b7dc8bdbe7be5d071325d09 -r 1eda20e058ebb08a4f4b71ed6988f6f106b62fdb lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -27,7 +27,7 @@
         we want here. We could add a parameter to use the recursive
         style, but this is meant to act similar to an "ls" directory listing.
         """
-        folder_contents = []
+        folder_container = []
         current_user_roles = trans.get_current_user_roles()
 
 
@@ -100,7 +100,8 @@
             
         # Return the reversed path so it starts with the library node.
         full_path = build_path( folder )[::-1]
-
+        folder_container.append( dict( full_path = full_path ) )
+        folder_contents = []
         # Go through every item in the folder and include its meta-data.
         for content_item in load_folder_contents( folder ):
             return_item = {}
@@ -118,13 +119,11 @@
             return_item.update( dict( id = encoded_id,
                                type = content_item.api_type,
                                name = content_item.name,
-                               library_id = encoded_parent_library_id,
-                               full_path = full_path,
-                               url = url_for( 'folder_contents', folder_id=encoded_id ) ) )
+                               library_id = encoded_parent_library_id
+                                ) )
             folder_contents.append( return_item )
-        if len( folder_contents ) == 0:
-            folder_contents.append( dict( full_path = full_path ) )
-        return folder_contents
+        folder_container.append( dict( folder_contents = folder_contents ) )
+        return folder_container
 
     @web.expose_api
     def show( self, trans, id, library_id, **kwd ):
diff -r db8fd059cf4cae9a6b7dc8bdbe7be5d071325d09 -r 1eda20e058ebb08a4f4b71ed6988f6f106b62fdb static/AAA_scratch/data_library/index.html
--- /dev/null
+++ b/static/AAA_scratch/data_library/index.html
@@ -0,0 +1,265 @@
+<!doctype html>
+<html lang="en">
+<head>
+  <meta charset="utf-8">
+  <title>Data Library</title>
+  <link rel="stylesheet" href="http://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/2.1.1/css/bootstrap…">
+</head>
+<body>
+
+
+  <div class="container">
+    <h1><a onclick="router.navigate('', {trigger: true})">Data Library</a></h1>
+    <hr />
+    <div class="page"></div>
+  </div>
+
+<!-- LIBRARY LIST TEMPLATE -->
+  <script type="text/template" id="library-list-template">
+    <table class="table striped">
+      <thead>
+        <th>name</th>
+        <th>description</th>
+        <th>synopsis</th> 
+        <th>model type</th> 
+        <th>id</th> 
+
+      </thead>
+      <tbody>
+        <% _.each(libraries, function(library) { %>
+          <tr>
+            <td><a href="#/folders/<%- library.id %>"><%- library.get('name') %></a></td>
+            <td><%= htmlEncode(library.get('description')) %></td>
+            <td><%= htmlEncode(library.get('synopsis')) %></td>
+            <td><%= htmlEncode(library.get('model_class')) %></td>
+            <td><a href="#/folders/<%- library.id %>"><%= htmlEncode(library.get('id')) %></a></td>
+          </tr>
+          <% }); %>
+      </tbody>
+    </table>
+  </script>
+
+
+<!-- FOLDERS LIST TEMPLATE -->
+  <script type="text/template" id="folder-content-template">
+    <table class="table striped">
+      <thead>
+        <th>name</th>
+        <th>type</th>
+        <th>parent id</th>
+        <th>id</th>
+        <th>library id</th>
+      </thead>
+      <tbody>
+        <% _.each(items, function(content_item) { %>
+          <tr>
+          <% if (content_item.get('type') === 'folder') { %>
+            <td><a href="#/folders/<%- content_item.id %>?library_id=<%- content_item.get('library_id') %>"><%- content_item.get('name') %></a></td>
+          <% } else {  %>
+            <td><%- content_item.get('name') %></td>
+          <% } %>    
+            <td><%= htmlEncode(content_item.get('type')) %></td>
+            <td><%= htmlEncode(content_item.get('parent_id')) %></td>
+            <td><a href="#/folders/<%- content_item.id %>"><%= htmlEncode(content_item.get('id')) %></a></td>
+            <td><a href="#/folders/<%- content_item.get('library_id') %>"><%= htmlEncode(content_item.get('library_id')) %></a></td>
+          </tr>
+          <% }); %>
+      </tbody>
+    </table>
+
+    <hr/>
+    <div>
+    <% if ( typeof libraries !== undefined ) { %>
+    <% _.each(libraries.models, function(library) { %>
+      <%- library.get('name') %><br/>
+      <% }); %>
+    <% } else { %>
+      libraries are undefined
+      <% } %>
+    </div>
+  </script>
+
+
+  <script src="jquery-1.10.2.js" type="text/javascript"></script>
+  <script src="underscore.js" type="text/javascript"></script>
+  <script src="backbone.js" type="text/javascript"></script>
+  <script src="backbone.queryparams.js" type="text/javascript"></script>
+
+
+  <script>
+
+    // ENCODE HTML
+    function htmlEncode(value){
+      return $('<div/>').text(value).html();
+    }
+
+    // NESTED COLLECTIONS
+    function nestCollection(model, attributeName, nestedCollection) {
+      //setup nested references
+      for (var i = 0; i < nestedCollection.length; i++) {
+        model.attributes[attributeName][i] = nestedCollection.at(i).attributes;
+      }
+
+      //create empty arrays if none
+      nestedCollection.bind('add', function (initiative) {
+        if (!model.get(attributeName)) {
+          model.attributes[attributeName] = [];
+        }
+        model.get(attributeName).push(initiative.attributes);
+      });
+
+      nestedCollection.bind('remove', function (initiative) {
+        var updateObj = {};
+        updateObj[attributeName] = _.without(model.get(attributeName), initiative.attributes);
+        model.set(updateObj);
+      });
+      return nestedCollection;
+    }
+
+    var api_key = 'e48b2b2589b242a50538f8f9fa8e5ee9';
+
+  // MODIFICATIONS DONE JUST BEFORE THE REQUEST ITSELF
+    $.ajaxPrefilter( function( options, originalOptions, jqXHR ) {
+      options.url = 'http://martenson.bx.psu.edu:8080' + options.url + '?key=' + api_key;
+    });
+
+  // SERIALIZE JSON
+    $.fn.serializeObject = function() {
+      var o = {};
+      var a = this.serializeArray();
+      $.each(a, function() {
+        if (o[this.name] !== undefined) {
+          if (!o[this.name].push) {
+            o[this.name] = [o[this.name]];
+          }
+          o[this.name].push(this.value || '');
+        } else {
+          o[this.name] = this.value || '';
+        }
+      });
+      return o;
+    };
+
+    // LIBRARY
+    var Library = Backbone.Model.extend({
+      urlRoot: '/api/libraries',
+    });
+    var Libraries = Backbone.Collection.extend({
+      url: '/api/libraries',
+      model: Library
+    });
+
+    // ITEM
+    var Item = Backbone.Model.extend({
+    })
+
+    // FOLDER
+    var Folder = Backbone.Collection.extend({
+      model: Item      
+    })
+
+    // Container including metadata
+    var FolderContainer = Backbone.Model.extend({
+       defaults: {
+        folder: new Folder(),
+        full_path: "unknown",
+        urlRoot: "/api/folders/",
+        id: "unknown"
+      },
+      parse: function(obj) {
+          full_path = obj[0].full_path;
+          // update the inner collection
+          this.get("folder").reset(obj[1].folder_contents);
+
+          // this mightn't be necessary
+          // delete obj.folder;
+
+          return obj;
+        }
+    })
+
+    var LibraryListView = Backbone.View.extend({
+      el: '.page',
+      render: function () {
+        var that = this;
+        if (typeof libraries === "undefined") {
+          libraries = new Libraries();
+        }
+        libraries.fetch({
+          success: function (libraries) {
+            var template = _.template($('#library-list-template').html(), {libraries: libraries.models});
+            that.$el.html(template);
+          }
+        })
+      }
+    });
+
+  
+    var FolderContentView = Backbone.View.extend({
+      el: '.page',
+      render: function (options, params) {
+        var that = this;
+
+        var folderContainer = new FolderContainer({id: options.id});
+        folderContainer.url = folderContainer.attributes.urlRoot + options.id + '/contents';
+        // var folder = new Folder({id: options.id});
+
+        folderContainer.fetch({
+          success: function (container) {
+          folderContainer.attributes.folder = container.attributes.folder;
+          var template = _.template($('#folder-content-template').html(), {items: folderContainer.attributes.folder.models, id: options.id});
+          that.$el.html(template);
+          }
+        })
+
+        // folder.fetch({
+        //   success: function (folder) {
+        //   folderContainer.folder = folder;
+        //   var template = _.template($('#folder-content-template').html(), {items: folderContainer.folder.models, id: options.id});
+        //   that.$el.html(template);
+        //   }
+        // })
+
+        // var folder = new Folder({id: options.id});
+        // folder.url = folder.urlRoot + options.id + '/contents';
+        // folder.fetch({
+        //   success: function (folder) {
+        //     // console.log(JSON.stringify(folder))
+        //     //create the template
+        //     var template = _.template($('#folder-content-template').html(), {items: folder.models, id: options.id});
+        //     that.$el.html(template);
+        //   }
+        // })
+
+      },
+    });
+
+
+    var libraryListView = new LibraryListView();
+    var folderContentView = new FolderContentView();
+
+    var Router = Backbone.Router.extend({
+      routes: {
+            "" :                                "libraries", 
+            "folders/:id" :                     "folder_content"
+          }
+    });
+
+    var router = new Router;
+
+    router.on('route:libraries', function() {
+      // render libraries list
+      libraryListView.render();
+    })
+
+    router.on('route:folder_content', function(id, params) {
+      // render folder's content
+      folderContentView.render({id: id, params: params});
+    })
+
+    Backbone.history.start();
+  </script>
+
+
+</body>
+</html> 
diff -r db8fd059cf4cae9a6b7dc8bdbe7be5d071325d09 -r 1eda20e058ebb08a4f4b71ed6988f6f106b62fdb static/AAA_scratch/data_library/index_working.html
--- /dev/null
+++ b/static/AAA_scratch/data_library/index_working.html
@@ -0,0 +1,261 @@
+<!doctype html>
+<html lang="en">
+<head>
+  <meta charset="utf-8">
+  <title>Data Library</title>
+  <link rel="stylesheet" href="http://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/2.1.1/css/bootstrap…">
+</head>
+<body>
+
+
+  <div class="container">
+    <h1>Data Library</h1>
+    <hr />
+    <div class="page"></div>
+  </div>
+
+<!-- LIBRARY LIST TEMPLATE -->
+  <script type="text/template" id="library-list-template">
+    <a href="#/new" class="btn btn-primary">New Library</a>
+    <hr />
+    <table class="table striped">
+      <thead>
+        <th>name</th>
+        <th>description</th>
+        <th>synopsis</th> 
+        <th></th>
+        <th></th>
+      </thead>
+      <tbody>
+        <% _.each(libraries, function(library) { %>
+          <tr>
+            <td><a href="#/folders/<%- library.id %>"><%- library.get('name') %></a></td>
+            <td><%= htmlEncode(library.get('description')) %></td>
+            <td><%= htmlEncode(library.get('synopsis')) %></td>
+            <td><a href="#/edit/<%= library.id %>" class="btn">Edit</td>
+            <td><button data-library-id="<%- library.id %>" class="btn btn-danger delete">Delete</button></td>
+          </tr>
+          <% }); %>
+      </tbody>
+    </table>
+  </script>
+
+<!-- LIBRARY EDIT/CREATE TEMPLATE   -->
+  <script type="text/template" id="library-edit-template">
+    <form class="library-edit-form">
+      <legend>Create Library</legend>
+      <label>Name of the Library</label>
+      <input type="text" name="name" />
+      <label>Description</label>
+      <input type="text" name="description" />
+      <label>Synopsis</label>
+      <input type="text" name="synopsis" />
+      <hr />
+      <button type="submit" class="btn">Create Library</button>
+    </form>
+  </script> 
+
+
+<!-- FOLDERS LIST TEMPLATE -->
+  <script type="text/template" id="folders-list-template">
+    <table class="table striped">
+      <thead>
+        <th>name</th>
+        <th>type</th>
+      </thead>
+      <tbody>
+        <% _.each(folders, function(folder) { %>
+          <tr>
+          <% if (folder.get('type') === 'folder') { %>
+            <td><a href="#/folders/<%- folder.id %>"><%- folder.get('name') %></a></td>
+          <% } else {  %>
+            <td><%- folder.get('name') %></td>
+          <% } %>    
+            <td><%= htmlEncode(folder.get('type')) %></td>
+          </tr>
+          <% }); %>
+      </tbody>
+    </table>
+  </script>
+
+
+  <script src="jquery-1.10.2.js" type="text/javascript"></script>
+  <script src="underscore.js" type="text/javascript"></script>
+  <script src="backbone.js" type="text/javascript"></script>
+
+
+  <script>
+
+  // ENCODE HTML
+    function htmlEncode(value){
+      return $('<div/>').text(value).html();
+    }
+
+    var api_key = 'e48b2b2589b242a50538f8f9fa8e5ee9';
+
+  // MODIFICATIONS DONE JUST BEFORE THE REQUEST ITSELF
+    $.ajaxPrefilter( function( options, originalOptions, jqXHR ) {
+      options.url = 'http://martenson.bx.psu.edu:8080' + options.url + '?key=' + api_key;
+    });
+
+  // SERIALIZE JSON
+    $.fn.serializeObject = function() {
+      var o = {};
+      var a = this.serializeArray();
+      $.each(a, function() {
+        if (o[this.name] !== undefined) {
+          if (!o[this.name].push) {
+            o[this.name] = [o[this.name]];
+          }
+          o[this.name].push(this.value || '');
+        } else {
+          o[this.name] = this.value || '';
+        }
+      });
+      return o;
+    };
+
+// LIBRARY
+    var Library = Backbone.Model.extend({
+      urlRoot: '/api/libraries'
+    });
+    var Libraries = Backbone.Collection.extend({
+      url: '/api/libraries'
+    });
+// FOLDER
+    var Folder = Backbone.Model.extend({
+      urlRoot: '/api/folders/'
+    })
+    var Folders = Backbone.Collection.extend({
+      model: Folder,
+      urlRoot: '/api/folders/'
+
+    })
+
+    var LibraryEditView = Backbone.View.extend({
+      el: '.page',
+      events: {
+        'submit .library-edit-form': 'saveLibrary'
+      },
+      saveLibrary: function (ev) {
+        var libraryDetails = $(ev.currentTarget).serializeObject();
+        var library = new Library();
+        library.save(libraryDetails, {
+          success: function (library) {
+            router.navigate('', {trigger: true})
+          }
+        });
+        console.log(libraryDetails);
+        return false;
+      },
+      render: function (options) {
+        var that = this;
+        if (options.id) {
+           this.library = new Library({id: options.id});
+            this.library.fetch({
+              success: function(library){
+                var template = _.template($('#library-edit-template').html(), {library: library});
+                that.$el.html(template);
+              }
+            });
+        } else {
+          var template = _.template($('#library-edit-template').html(), {user: null});
+          this.$el.html(template);
+        }
+      }
+    });
+
+
+    var LibraryListView = Backbone.View.extend({
+      el: '.page',
+      events: {
+        'click .delete' : 'deleteLibrary'
+      },
+      render: function () {
+        var that = this;
+        var libraries = new Libraries();
+        libraries.fetch({
+          success: function (libraries) {
+            var template = _.template($('#library-list-template').html(), {libraries: libraries.models});
+            that.$el.html(template);
+          }
+        })
+      },
+      deleteLibrary : function (ev) {
+        // DELETE /libraries/{id}
+        var id_to_delete = $(ev.target).attr('data-library-id');
+        var libraryToDelete = new Library({id: id_to_delete})
+        libraryToDelete.destroy({
+          success: function () {
+            libraryListView.render();
+          }
+        })
+        return false
+      },
+      createLibrary : function (ev) {
+        // PUT /libraries
+
+      }
+    });
+
+  
+    var FoldersListView = Backbone.View.extend({
+      el: '.page',
+      render: function (options) {
+        var that = this;
+        var folders = new Folders({id: options.id});
+        folders.url = folders.urlRoot + options.id + '/contents';
+        folders.fetch({
+          success: function (folders) {
+            var template = _.template($('#folders-list-template').html(), {folders: folders.models, id: options.id});
+            that.$el.html(template);
+          }
+        })
+      },
+    });
+
+
+    var libraryListView = new LibraryListView();
+    var libraryEditView = new LibraryEditView();
+    var foldersListView = new FoldersListView();
+
+
+    var Router = Backbone.Router.extend({
+      routes: {
+            "" :                                "home", 
+            "new" :                             "edit_library",
+            "edit/:id" :                        "edit_library",
+            "delete/:id" :                      "delete_library",
+            "folders/:id" :                     "folders"
+          }
+    });
+
+    var router = new Router;
+
+    router.on('route:home', function() {
+      // render libraries list
+      libraryListView.render();
+    })
+    router.on('route:edit_library', function(id) {
+      // render library edit form
+      libraryEditView.render({id: id});
+    })
+    router.on('route:delete_library', function() {
+      // render delete library form
+      libraryDeleteView.render();
+    }) 
+    // router.on('route:folders', function(parent_id, top_level) {
+    //   // render folders list
+    //   foldersListView.render({id: parent_id, top_level: true});
+    // })    
+    router.on('route:folders', function(id) {
+      // render folders list
+      foldersListView.render({id: id});
+    })
+
+    Backbone.history.start();
+  </script>
+
+
+</body>
+</html> 
https://bitbucket.org/galaxy/galaxy-central/commits/dc735a38cf76/
Changeset:   dc735a38cf76
Branch:      data_library
User:        martenson
Date:        2013-10-24 20:22:26
Summary:     refactoring, modals start
Affected #:  4 files
diff -r 1eda20e058ebb08a4f4b71ed6988f6f106b62fdb -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 lib/galaxy/model/__init__.py
--- a/lib/galaxy/model/__init__.py
+++ b/lib/galaxy/model/__init__.py
@@ -1835,7 +1835,7 @@
         return name
 
 class LibraryFolder( object, Dictifiable ):
-    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build' )
+    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build', 'update_time' )
     def __init__( self, name=None, description=None, item_count=0, order_id=None ):
         self.name = name or "Unnamed folder"
         self.description = description
diff -r 1eda20e058ebb08a4f4b71ed6988f6f106b62fdb -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -15,42 +15,37 @@
     Class controls retrieval, creation and updating of folder contents.
     """
 
+    def load_folder_contents( self, trans, folder ):
+        """
+        Loads all contents of the folder (folders and data sets) but only in the first level.
+        """
+        current_user_roles = trans.get_current_user_roles()
+        is_admin = trans.user_is_admin()
+        content_items = []
+        for subfolder in folder.active_folders:
+            if not is_admin:
+                can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
+            if (is_admin or can_access) and not subfolder.deleted:
+                subfolder.api_type = 'folder'
+                content_items.append( subfolder )
+        for dataset in folder.datasets:
+            if not is_admin:
+                can_access = trans.app.security_agent.can_access_dataset( current_user_roles, dataset.library_dataset_dataset_association.dataset )
+            if (is_admin or can_access) and not dataset.deleted:
+                dataset.api_type = 'file'
+                content_items.append( dataset )
+        return content_items
+
     @web.expose_api
     def index( self, trans, folder_id, **kwd ):
         """
         GET /api/folders/{encoded_folder_id}/contents
-        
         Displays a collection (list) of a folder's contents (files and folders).
-        
-        The /api/library_contents/{encoded_library_id}/contents
-        lists everything in a library recursively, which is not what
-        we want here. We could add a parameter to use the recursive
-        style, but this is meant to act similar to an "ls" directory listing.
+        Encoded folder ID is prepended with 'F' if it is a folder as opposed to a data set which does not have it.
         """
         folder_container = []
         current_user_roles = trans.get_current_user_roles()
 
-
-        def load_folder_contents( folder ):
-            """
-            Load contents of the folder (folders and datasets).
-            """
-            admin = trans.user_is_admin()
-            rval = []
-            for subfolder in folder.active_folders:
-                if not admin:
-                    can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
-                if (admin or can_access) and not subfolder.deleted:
-                    subfolder.api_type = 'folder'
-                    rval.append( subfolder )
-            for ld in folder.datasets:
-                if not admin:
-                    can_access = trans.app.security_agent.can_access_dataset( current_user_roles, ld.library_dataset_dataset_association.dataset )
-                if (admin or can_access) and not ld.deleted:
-                    ld.api_type = 'file'
-                    rval.append( ld )
-            return rval
-
         try:
             decoded_folder_id = trans.security.decode_id( folder_id[-16:] )
         except TypeError:
@@ -59,69 +54,68 @@
 
         try:
             folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( decoded_folder_id )
-#             log.debug("XXXXXXXXXXXXXXXXXXXXXXXXXXX folder.parent_library" + str(folder.parent_library.id))
-#             log.debug("XXXXXXXXXXXXXXXXXXXXXXXXXXX folder.parent_id" + str(folder.parent_id))
             parent_library = folder.parent_library
         except:
             folder = None
-            log.error( "FolderContentsController.index: Unable to retrieve folder %s"
-                      % folder_id )
+            log.error( "FolderContentsController.index: Unable to retrieve folder with ID: %s" % folder_id )
 
-        # TODO: Find the API's path to this folder if necessary.
-        # This was needed in recursive descent, but it's not needed
-        # for "ls"-style content checking:
-        if not folder or not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+        # We didn't find the folder or user does not have an access to it.
+        if not folder:
             trans.response.status = 400
             return "Invalid folder id ( %s ) specified." % str( folder_id )
-        # TODO MARTEN Can it be that predecessors of current folder have different access rights? aka user shouldn't see them?
         
-        # Search the path upwards and load the whole route of names and ids for breadcrumb purposes.
+        if not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+            log.warning( "SECURITY: User (id: %s) without proper access rights is trying to load folder with ID of %s" % ( trans.user.id, folder.id ) )
+            trans.response.status = 400
+            return "Invalid folder id ( %s ) specified." % str( folder_id )
+        
         path_to_root = []
-        
         def build_path ( folder ):
+            """
+            Search the path upwards recursively and load the whole route of names and ids for breadcrumb purposes.
+            """
             path_to_root = []
             # We are almost in root
-            log.debug( "XXXXXXXXXXXXXXXXXXXXXXX folder.parent_id: " + str( folder.parent_id ) )
-            log.debug( "XXXXXXXXXXXXXXXXXXXXXXX folder.parent_library.id: " + str( folder.parent_library.id ) )
             if folder.parent_id is None:
                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ALMOST ROOT FOLDER! ADDING: " + str( folder.name ) )
-                path_to_root.append( ( folder.id, folder.name ) )
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
 #                 upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_library.id )
 #                 path_to_root.append( ( upper_folder.id, upper_folder.name ) )
             else:
             # We add the current folder and traverse up one folder.
                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ADDING THIS FOLDER AND TRAVERSING UP:  " + str( folder.name ) )
-                path_to_root.append( ( folder.id, folder.name ) )
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
                 upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_id )
                 path_to_root.extend( build_path( upper_folder ) )
-#             path_to_root = path_to_root.reverse()
-#             return path_to_root[::-1]
             return path_to_root
             
         # Return the reversed path so it starts with the library node.
         full_path = build_path( folder )[::-1]
         folder_container.append( dict( full_path = full_path ) )
+        
         folder_contents = []
         # Go through every item in the folder and include its meta-data.
-        for content_item in load_folder_contents( folder ):
+        for content_item in self.load_folder_contents( trans, folder ):
             return_item = {}
             encoded_id = trans.security.encode_id( content_item.id )
             
             # For folder return also hierarchy values
             if content_item.api_type == 'folder':
-                encoded_parent_library_id = trans.security.encode_id( content_item.parent_library.id )
+#                 encoded_parent_library_id = trans.security.encode_id( content_item.parent_library.id )
                 encoded_id = 'F' + encoded_id
-                if content_item.parent_id is not None: # Return folder's parent id for browsing back.
-                    encoded_parent_id = 'F' + trans.security.encode_id( content_item.parent_id )
-                    return_item.update ( dict ( parent_id = encoded_parent_id ) )
+#                 if content_item.parent_id is not None: # Return folder's parent id for browsing back.
+#                     encoded_parent_id = 'F' + trans.security.encode_id( content_item.parent_id )
+                last_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+                return_item.update ( dict ( item_count = content_item.item_count, last_updated = last_updated ) )
             
             # For every item return also the default meta-data
             return_item.update( dict( id = encoded_id,
                                type = content_item.api_type,
-                               name = content_item.name,
-                               library_id = encoded_parent_library_id
+                               name = content_item.name
+                               
                                 ) )
             folder_contents.append( return_item )
+        # Put the data in the container
         folder_container.append( dict( folder_contents = folder_contents ) )
         return folder_container
 
diff -r 1eda20e058ebb08a4f4b71ed6988f6f106b62fdb -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 lib/galaxy/webapps/galaxy/api/libraries.py
--- a/lib/galaxy/webapps/galaxy/api/libraries.py
+++ b/lib/galaxy/webapps/galaxy/api/libraries.py
@@ -51,7 +51,7 @@
         for library in query:
             item = library.to_dict( view='element' )
             item['url'] = url_for( route, id=trans.security.encode_id( library.id ) )
-            item['id'] = trans.security.encode_id( item['id'] )
+            item['id'] = 'F' + trans.security.encode_id( item['id'] )
             rval.append( item )
         return rval
 
diff -r 1eda20e058ebb08a4f4b71ed6988f6f106b62fdb -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 static/AAA_scratch/data_library/index.html
--- a/static/AAA_scratch/data_library/index.html
+++ b/static/AAA_scratch/data_library/index.html
@@ -1,10 +1,13 @@
-<!doctype html>
+<!DOCTYPE html><html lang="en"><head><meta charset="utf-8"><title>Data Library</title>
-  <link rel="stylesheet" href="http://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/2.1.1/css/bootstrap…">
+  <meta name="viewport" content="width=device-width, initial-scale=1.0">
+  <!-- Bootstrap -->
+  <link href="css/bootstrap.min.css" rel="stylesheet" media="screen"></head>
+
 <body>
 
 
@@ -16,7 +19,7 @@
 
 <!-- LIBRARY LIST TEMPLATE --><script type="text/template" id="library-list-template">
-    <table class="table striped">
+    <table class="table table-striped"><thead><th>name</th><th>description</th>
@@ -42,45 +45,47 @@
 
 <!-- FOLDERS LIST TEMPLATE --><script type="text/template" id="folder-content-template">
-    <table class="table striped">
+
+          <% _.each(path, function(path_item) { %>
+            <% if (path_item[0] != id) { %>
+              <a href='#/folders/<%- path_item[0] %>'><%- path_item[1] %></a> |
+            <% } else { %>
+              <%- path_item[1] %>
+            <% } %>
+          <% }); %>
+
+    <table class="table table-hover table-condensed"><thead>
+        <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th><th>name</th><th>type</th>
-        <th>parent id</th>
-        <th>id</th>
-        <th>library id</th></thead><tbody><% _.each(items, function(content_item) { %>
-          <tr>
+          <tr class="folder_row" id="<%- content_item.id %>">
+            <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td><% if (content_item.get('type') === 'folder') { %>
-            <td><a href="#/folders/<%- content_item.id %>?library_id=<%- content_item.get('library_id') %>"><%- content_item.get('name') %></a></td>
+            <td><a href="#/folders/<%- content_item.id %>"><%- content_item.get('name') %></a>
+            <% if (content_item.get('item_count') === 0) { %>
+              <span class="muted">(empty folder)</span>
+            <% } %>
+            </td><% } else {  %><td><%- content_item.get('name') %></td><% } %><td><%= htmlEncode(content_item.get('type')) %></td>
-            <td><%= htmlEncode(content_item.get('parent_id')) %></td>
-            <td><a href="#/folders/<%- content_item.id %>"><%= htmlEncode(content_item.get('id')) %></a></td>
-            <td><a href="#/folders/<%- content_item.get('library_id') %>"><%= htmlEncode(content_item.get('library_id')) %></a></td></tr><% }); %></tbody></table>
 
-    <hr/>
-    <div>
-    <% if ( typeof libraries !== undefined ) { %>
-    <% _.each(libraries.models, function(library) { %>
-      <%- library.get('name') %><br/>
-      <% }); %>
-    <% } else { %>
-      libraries are undefined
-      <% } %>
-    </div>
+
   </script>
 
 
+  <script src="require.js"></script><script src="jquery-1.10.2.js" type="text/javascript"></script>
+  <script src="js/bootstrap.min.js"></script><script src="underscore.js" type="text/javascript"></script><script src="backbone.js" type="text/javascript"></script><script src="backbone.queryparams.js" type="text/javascript"></script>
@@ -88,34 +93,27 @@
 
   <script>
 
+ // configure require
+  //      require.config({
+  //          baseUrl: "/",
+  //          shim: {
+  //              "libs/underscore": { exports: "_" },
+  //              "libs/backbone/backbone": { exports: "Backbone" },
+  //              "libs/backbone/backbone-relational": ["libs/backbone/backbone"]
+  //          }
+  //      });
+
+  // require(['galaxy.modal'], function(modal)
+  //       {
+  //           Galaxy.modal = new modal.GalaxyModal();
+  //       });
+
+
     // ENCODE HTML
     function htmlEncode(value){
       return $('<div/>').text(value).html();
     }
 
-    // NESTED COLLECTIONS
-    function nestCollection(model, attributeName, nestedCollection) {
-      //setup nested references
-      for (var i = 0; i < nestedCollection.length; i++) {
-        model.attributes[attributeName][i] = nestedCollection.at(i).attributes;
-      }
-
-      //create empty arrays if none
-      nestedCollection.bind('add', function (initiative) {
-        if (!model.get(attributeName)) {
-          model.attributes[attributeName] = [];
-        }
-        model.get(attributeName).push(initiative.attributes);
-      });
-
-      nestedCollection.bind('remove', function (initiative) {
-        var updateObj = {};
-        updateObj[attributeName] = _.without(model.get(attributeName), initiative.attributes);
-        model.set(updateObj);
-      });
-      return nestedCollection;
-    }
-
     var api_key = 'e48b2b2589b242a50538f8f9fa8e5ee9';
 
   // MODIFICATIONS DONE JUST BEFORE THE REQUEST ITSELF
@@ -167,13 +165,10 @@
         id: "unknown"
       },
       parse: function(obj) {
-          full_path = obj[0].full_path;
+          this.full_path = obj[0].full_path;
+          // this.folder.reset(obj[1].folder_contents);
           // update the inner collection
           this.get("folder").reset(obj[1].folder_contents);
-
-          // this mightn't be necessary
-          // delete obj.folder;
-
           return obj;
         }
     })
@@ -197,41 +192,39 @@
   
     var FolderContentView = Backbone.View.extend({
       el: '.page',
+      events: {
+        'click #select-all-checkboxes' : 'selectAll',
+        'click .folder_row' : 'selectClicked'
+      },
       render: function (options, params) {
         var that = this;
 
         var folderContainer = new FolderContainer({id: options.id});
         folderContainer.url = folderContainer.attributes.urlRoot + options.id + '/contents';
-        // var folder = new Folder({id: options.id});
 
         folderContainer.fetch({
           success: function (container) {
-          folderContainer.attributes.folder = container.attributes.folder;
-          var template = _.template($('#folder-content-template').html(), {items: folderContainer.attributes.folder.models, id: options.id});
+          // folderContainer.attributes.folder = container.attributes.folder;
+          var template = _.template($('#folder-content-template').html(), {path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id});
           that.$el.html(template);
           }
         })
-
-        // folder.fetch({
-        //   success: function (folder) {
-        //   folderContainer.folder = folder;
-        //   var template = _.template($('#folder-content-template').html(), {items: folderContainer.folder.models, id: options.id});
-        //   that.$el.html(template);
-        //   }
-        // })
-
-        // var folder = new Folder({id: options.id});
-        // folder.url = folder.urlRoot + options.id + '/contents';
-        // folder.fetch({
-        //   success: function (folder) {
-        //     // console.log(JSON.stringify(folder))
-        //     //create the template
-        //     var template = _.template($('#folder-content-template').html(), {items: folder.models, id: options.id});
-        //     that.$el.html(template);
-        //   }
-        // })
-
       },
+      selectAll : function (ev) {
+         var selected = ev.target.checked;
+         // Iterate each checkbox
+         $(':checkbox').each(function () { this.checked = selected; });
+      },
+      selectClicked : function (ev) {
+        var checkbox = $("#" + ev.target.parentElement.id).find(':checkbox')
+        if (checkbox[0] != undefined) {
+          if (checkbox[0].checked){
+            checkbox[0].checked = '';
+          } else {
+            checkbox[0].checked = 'selected';
+          }
+        }
+      }
     });
 
 
@@ -258,8 +251,7 @@
     })
 
     Backbone.history.start();
+
   </script>
-
-
 </body></html> 
https://bitbucket.org/galaxy/galaxy-central/commits/518ded2c3f30/
Changeset:   518ded2c3f30
Branch:      data_library
User:        martenson
Date:        2013-11-06 23:15:41
Summary:     work in progress on data libraries, integrated within galaxy framework with require, modularized
Affected #:  7 files
diff -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 -r 518ded2c3f30cca803a29f76da2e80505b166647 lib/galaxy/webapps/galaxy/controllers/library.py
--- a/lib/galaxy/webapps/galaxy/controllers/library.py
+++ b/lib/galaxy/webapps/galaxy/controllers/library.py
@@ -76,6 +76,22 @@
 
     library_list_grid = LibraryListGrid()
 
+    
+    @web.expose
+    def list( self, trans, **kwd ):
+        params = util.Params( kwd )
+
+        # define app configuration for generic mako template
+        app = {
+            'jscript'       : "galaxy.library"
+        }
+
+        # fill template
+        return trans.fill_template('galaxy.panels.mako', config = {'app' : app})
+    
+    
+#         return trans.fill_template( "/library/list.mako")
+    
     @web.expose
     def index( self, trans, **kwd ):
         params = util.Params( kwd )
diff -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 -r 518ded2c3f30cca803a29f76da2e80505b166647 static/AAA_scratch/data_library/index.html
--- a/static/AAA_scratch/data_library/index.html
+++ b/static/AAA_scratch/data_library/index.html
@@ -83,7 +83,7 @@
   </script>
 
 
-  <script src="require.js"></script>
+
   <script src="jquery-1.10.2.js" type="text/javascript"></script><script src="js/bootstrap.min.js"></script><script src="underscore.js" type="text/javascript"></script>
@@ -93,22 +93,6 @@
 
   <script>
 
- // configure require
-  //      require.config({
-  //          baseUrl: "/",
-  //          shim: {
-  //              "libs/underscore": { exports: "_" },
-  //              "libs/backbone/backbone": { exports: "Backbone" },
-  //              "libs/backbone/backbone-relational": ["libs/backbone/backbone"]
-  //          }
-  //      });
-
-  // require(['galaxy.modal'], function(modal)
-  //       {
-  //           Galaxy.modal = new modal.GalaxyModal();
-  //       });
-
-
     // ENCODE HTML
     function htmlEncode(value){
       return $('<div/>').text(value).html();
@@ -245,9 +229,9 @@
       libraryListView.render();
     })
 
-    router.on('route:folder_content', function(id, params) {
+    router.on('route:folder_content', function(id) {
       // render folder's content
-      folderContentView.render({id: id, params: params});
+      folderContentView.render({id: id});
     })
 
     Backbone.history.start();
diff -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 -r 518ded2c3f30cca803a29f76da2e80505b166647 static/scripts/galaxy.library.js
--- /dev/null
+++ b/static/scripts/galaxy.library.js
@@ -0,0 +1,333 @@
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+// === GALAXY LIBRARY MODULE ====
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+
+// global variables
+var view            = null;
+var library_router  = null;
+
+// dependencies
+define(["galaxy.modal", "galaxy.master", "galaxy.library.router"], function(mod_modal, mod_master, router) {
+
+// MMMMMMMMMMMMMMM
+// === Models ====
+// MMMMMMMMMMMMMMM
+
+    // LIBRARY
+    var Library = Backbone.Model.extend({
+      urlRoot: '/api/libraries',
+    });
+
+    // LIBRARIES
+    var Libraries = Backbone.Collection.extend({
+      url: '/api/libraries',
+      model: Library
+    });
+
+    // ITEM
+    var Item = Backbone.Model.extend({
+    })
+
+    // FOLDER
+    var Folder = Backbone.Collection.extend({
+      model: Item      
+    })
+
+    // Container for folder contents (folders, items and metadata).
+    var FolderContainer = Backbone.Model.extend({
+       defaults: {
+        folder: new Folder(),
+        full_path: "unknown",
+        urlRoot: "/api/folders/",
+        id: "unknown"
+      },
+      parse: function(obj) {
+          this.full_path = obj[0].full_path;
+          // this.folder.reset(obj[1].folder_contents);
+          // update the inner collection
+          this.get("folder").reset(obj[1].folder_contents);
+          return obj;
+        }
+    })
+
+// MMMMMMMMMMMMMM
+// === Views ====
+// MMMMMMMMMMMMMM
+
+// galaxy folder
+var FolderContentView = Backbone.View.extend({
+      el : '#center',
+    // initialize
+    initialize : function(){
+        // view = this;
+        //set up the library router
+        // this.set_up_router();
+
+        //render
+        // this.render();
+    },    
+   // set up router
+    set_up_router : function(){   
+        if (library_router === null){
+            library_router = new router.LibraryRouter();
+            Backbone.history.start();   
+        }
+    },
+      template_folder : function (){
+                tmpl_array = [];
+
+                tmpl_array.push('<% _.each(path, function(path_item) { %>');
+                tmpl_array.push('<% if (path_item[0] != id) { %>');
+                tmpl_array.push('<a href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a> |');
+                tmpl_array.push('<% } else { %>');
+                tmpl_array.push('<%- path_item[1] %>');
+                tmpl_array.push('<% } %>');
+                tmpl_array.push('<% }); %>');
+                tmpl_array.push('<table class="table table-hover table-condensed">');
+                tmpl_array.push('   <thead>');
+                tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
+                tmpl_array.push('       <th>name</th>');
+                tmpl_array.push('       <th>type</th>');
+                tmpl_array.push('   </thead>');
+                tmpl_array.push('   <tbody>');
+                tmpl_array.push('       <% _.each(items, function(content_item) { %>');
+                tmpl_array.push('       <tr class="folder_row" id="<%- content_item.id %>">');
+                tmpl_array.push('           <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
+                tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>');
+                tmpl_array.push('           <td><a href="#/folders/<%- content_item.id %>"><%- content_item.get("name") %></a>');
+                tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>');
+                tmpl_array.push('           <span class="muted">(empty folder)</span>');
+                tmpl_array.push('           <% } %>');
+                tmpl_array.push('           </td>');
+                tmpl_array.push('           <% } else {  %>');
+                tmpl_array.push('           <td><%- content_item.get("name") %></td>');
+                tmpl_array.push('           <% } %>  ');
+                tmpl_array.push('           <td><%= _.escape(content_item.get("type")) %></td>');
+                tmpl_array.push('       </tr>');
+                tmpl_array.push('       <% }); %>');
+                tmpl_array.push('       ');
+                tmpl_array.push('       ');
+                tmpl_array.push('       ');
+                tmpl_array.push('   </tbody>');
+                tmpl_array.push('</table>');
+
+                return tmpl_array.join('');
+      },
+      events: {
+        'click #select-all-checkboxes' : 'selectAll',
+        'click .folder_row' : 'selectClicked'
+      },
+      render: function (options) {
+        var that = this;
+
+        var folderContainer = new FolderContainer({id: options.id});
+        folderContainer.url = folderContainer.attributes.urlRoot + options.id + '/contents';
+
+        folderContainer.fetch({
+          success: function (container) {
+          // folderContainer.attributes.folder = container.attributes.folder;
+          var template = _.template(that.template_folder(), {path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id});
+          that.$el.html(template);
+          }
+        })
+      },
+      selectAll : function (ev) {
+         var selected = ev.target.checked;
+         // Iterate each checkbox
+         $(':checkbox').each(function () { this.checked = selected; });
+      },
+      selectClicked : function (ev) {
+        var checkbox = $("#" + ev.target.parentElement.id).find(':checkbox')
+        if (checkbox[0] != undefined) {
+          if (checkbox[0].checked){
+            checkbox[0].checked = '';
+          } else {
+            checkbox[0].checked = 'selected';
+          }
+        }
+      }
+});
+
+
+// galaxy library
+var GalaxyLibraryview = Backbone.View.extend({
+    el: '#center',
+
+   events: {
+        'click #create_new_library_btn' : 'show_library_modal'
+      },
+
+       
+    // initialize
+    initialize : function(){
+        // view = this;
+        //set up the libray router
+        // this.set_up_router();
+
+        //render
+        // this.render();
+    },    
+
+    // set up router
+    set_up_router : function(){
+        if (library_router === null){
+            library_router = new router.LibraryRouter();
+            Backbone.history.start();   
+        }
+    },
+
+    // template
+    template_library_list : function (){
+        tmpl_array = [];
+
+        tmpl_array.push('');
+        tmpl_array.push('<h1>Welcome to the data libraries</h1>');
+        tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary">New Library</a>');
+        tmpl_array.push('<table class="table table-striped">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('    <th>name</th>');
+        tmpl_array.push('     <th>description</th>');
+        tmpl_array.push('     <th>synopsis</th> ');
+        tmpl_array.push('     <th>model type</th> ');
+        tmpl_array.push('     <th>id</th> ');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <% _.each(libraries, function(library) { %>');
+        tmpl_array.push('           <tr>');
+        tmpl_array.push('               <td><a href="#/folders/<%- library.id %>"><%- library.get("name") %></a></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
+        tmpl_array.push('               <td><a href="#/folders/<%- library.id %>"><%= _.escape(library.get("id")) %></a></td>');
+        tmpl_array.push('           </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('');
+        tmpl_array.push('');
+        tmpl_array.push('');
+        tmpl_array.push('');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+
+        return tmpl_array.join('');
+    },
+
+    // render
+    render: function () {
+        var that = this;
+        // if (typeof libraries === "undefined") {
+          libraries = new Libraries();
+        // }
+        libraries.fetch({
+          success: function (libraries) {
+            var template = _.template(that.template_library_list(), {libraries: libraries.models});
+            that.$el.html(template);
+          }
+        })
+    },
+
+    // own modal
+    modal : null,
+
+    // show/hide create library modal
+    show_library_modal : function (e)
+    {
+        // prevent default
+        e.preventDefault();
+        
+        // create modal
+        if (!this.modal){
+            // make modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal(
+            {
+                title   : 'Create New Library',
+                body    : this.template_new_library(),
+                buttons : {
+                    'Create' : function() {self.create_new_library_event()},
+                    'Close'  : function() {self.modal.hide()}
+                }
+            });
+        }
+        
+        // show modal
+        this.modal.show();
+    },
+    create_new_library_event: function(){
+        var libraryDetails = this.serialize_new_library();
+        var library = new Library();
+        library.save(libraryDetails, {
+          success: function (library) {
+
+            library_router.navigate('', {trigger: true})
+          }
+        });
+        // console.log(libraryDetails);
+        return false;
+    },
+    serialize_new_library : function(){
+        return {
+            name: $("input[name='Name']").val(),
+            description: $("input[name='Description']").val(),
+            synopsis: $("input[name='Synopsis']").val()
+        };
+    },
+    // load html template
+    template_new_library: function()
+    {
+        tmpl_array = [];
+        tmpl_array.push('<div id="new_library_modal">');
+        tmpl_array.push('<form>');
+        tmpl_array.push('<input type="text" name="Name" value="" placeholder="Name">');
+        tmpl_array.push('<input type="text" name="Description" value="" placeholder="Description">');
+        // tmpl_array.push('<label>Synopsis</label>');
+        tmpl_array.push('<input type="text" name="Synopsis" value="" placeholder="Synopsis">');
+        tmpl_array.push('');
+        tmpl_array.push('</form>');
+        tmpl_array.push('</div>');
+        return tmpl_array.join('');
+
+
+        // return  '<div id="'+ id +'"></div>';
+    }
+});
+
+// galaxy library wrapper View
+var GalaxyLibrary = Backbone.View.extend({
+    folderContentView : null,
+    galaxyLibraryview : null,
+    initialize : function(){
+
+        folderContentView = new FolderContentView();
+        galaxyLibraryview = new GalaxyLibraryview();
+
+        library_router = new router.LibraryRouter();
+            
+        library_router.on('route:libraries', function() {
+          // render libraries list
+          galaxyLibraryview.render();
+        })           
+         
+        library_router.on('route:folder_content', function(id) {
+          // render folder's contents
+          folderContentView.render({id: id});
+        
+        })      
+
+        // library_router.on('route:show_library_modal', function() {
+        //   // render folder's contents
+        //   galaxyLibraryview.show_library_modal();
+        
+        // })
+
+        Backbone.history.start();   
+
+        return this
+    }
+});
+
+// return
+return {
+    GalaxyApp: GalaxyLibrary
+};
+
+});
diff -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 -r 518ded2c3f30cca803a29f76da2e80505b166647 static/scripts/galaxy.library.router.js
--- /dev/null
+++ b/static/scripts/galaxy.library.router.js
@@ -0,0 +1,21 @@
+define( [], function() {
+
+/**
+ * -- Routers --
+ */
+
+/**
+ * Router for library browser.
+ */
+var LibraryRouter = Backbone.Router.extend({    
+    routes: {
+        "" :                 "libraries",
+        "folders/:id" :      "folder_content"
+    }
+});
+
+return {
+    LibraryRouter: LibraryRouter,
+};
+
+})
\ No newline at end of file
diff -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 -r 518ded2c3f30cca803a29f76da2e80505b166647 static/style/Gruntfile.js
--- a/static/style/Gruntfile.js
+++ b/static/style/Gruntfile.js
@@ -6,7 +6,7 @@
   var theme = grunt.option( 'theme', 'blue' );
 
   var out = 'blue'
-  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'trackster' ];
+  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'new_library', 'trackster' ];
 
   var _ = grunt.util._; 
   var fmt = _.sprintf; 
diff -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 -r 518ded2c3f30cca803a29f76da2e80505b166647 templates/base/base_panels.mako
--- a/templates/base/base_panels.mako
+++ b/templates/base/base_panels.mako
@@ -102,12 +102,13 @@
         });
         
         ## load galaxy js-modules
-        require(['galaxy.master', 'galaxy.frame', 'galaxy.modal', 'galaxy.upload'], function(master, frame, modal, upload)
+        require(['galaxy.master', 'galaxy.frame', 'galaxy.modal', 'galaxy.upload', 'galaxy.library'], function(master, frame, modal, upload, library)
         {
             Galaxy.master = new master.GalaxyMaster();
             Galaxy.frame_manager = new frame.GalaxyFrameManager();
             Galaxy.modal = new modal.GalaxyModal();
-            ##Galaxy.upload = new upload.GalaxyUpload();
+            Galaxy.upload = new upload.GalaxyUpload();
+            ##Galaxy.library = new library.GalaxyLibrary();
         });
     </script></%def>
diff -r dc735a38cf76f93772027b34e489db1ef9d2d0a5 -r 518ded2c3f30cca803a29f76da2e80505b166647 templates/webapps/galaxy/base_panels.mako
--- a/templates/webapps/galaxy/base_panels.mako
+++ b/templates/webapps/galaxy/base_panels.mako
@@ -126,6 +126,7 @@
     ## 'Shared Items' or Libraries tab.
     <%
         menu_options = [ 
+                        [ _('New Libraries'), h.url_for( controller='/library', action='list') ],
                         [ _('Data Libraries'), h.url_for( controller='/library', action='index') ],
                         None,
                         [ _('Published Histories'), h.url_for( controller='/history', action='list_published' ) ],
https://bitbucket.org/galaxy/galaxy-central/commits/3ea8be4161ab/
Changeset:   3ea8be4161ab
Branch:      data_library
User:        martenson
Date:        2013-11-12 22:48:56
Summary:     transfer into modal window, requirejs, notifications, import to history, galaxy.modal changes, API for library datasets, start of API versioning
Affected #:  6 files
diff -r 518ded2c3f30cca803a29f76da2e80505b166647 -r 3ea8be4161ab0a9ed2e1c17160c9d3d52d41b555 lib/galaxy/model/__init__.py
--- a/lib/galaxy/model/__init__.py
+++ b/lib/galaxy/model/__init__.py
@@ -2000,6 +2000,7 @@
                      genome_build = ldda.dbkey,
                      misc_info = ldda.info,
                      misc_blurb = ldda.blurb,
+                     peek = ( lambda ldda: ldda.display_peek() if ldda.peek and ldda.peek != 'no peek' else None )( ldda ),
                      template_data = template_data )
         if ldda.dataset.uuid is None:
             rval['uuid'] = None
diff -r 518ded2c3f30cca803a29f76da2e80505b166647 -r 3ea8be4161ab0a9ed2e1c17160c9d3d52d41b555 lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- /dev/null
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -0,0 +1,38 @@
+"""
+API operations on the contents of a dataset from library.
+"""
+from galaxy import web
+from galaxy.web.base.controller import BaseAPIController, UsesLibraryMixinItems
+
+import logging
+log = logging.getLogger( __name__ )
+
+class DatasetsController( BaseAPIController, UsesLibraryMixinItems ):
+
+    @web.expose_api
+    def index( self, trans, **kwd ):
+        """
+        GET /api/libraries/datasets
+        """
+        trans.response.status = 501
+        return 'not implemented'
+
+    @web.expose_api
+    def show( self, trans, id, **kwd ):
+        """
+        GET /api/libraries/datasets/{encoded_dataset_id}
+        Displays information about and/or content of a dataset identified by the lda ID.
+        """
+        # Get dataset.
+        try:
+            dataset = self.get_library_dataset( trans, id = id )
+        except Exception, e:
+            return str( e )
+        try:
+            # Default: return dataset as dict.
+            rval = dataset.to_dict()
+        except Exception, e:
+            rval = "Error in dataset API at listing contents: " + str( e )
+            log.error( rval + ": %s" % str(e), exc_info=True )
+            trans.response.status = 500
+        return rval
\ No newline at end of file
diff -r 518ded2c3f30cca803a29f76da2e80505b166647 -r 3ea8be4161ab0a9ed2e1c17160c9d3d52d41b555 lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -46,18 +46,19 @@
     atexit.register( app.shutdown )
     # Create the universe WSGI application
     webapp = GalaxyWebApplication( app, session_cookie='galaxysession', name='galaxy' )
-    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
-    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
-                      controller='admin_toolshed',
-                      action='display_image_in_repository',
-                      repository_id=None,
-                      image_file=None )
+
+    # Add the controllers folder
     webapp.add_ui_controllers( 'galaxy.webapps.galaxy.controllers', app )
+    # Add the api folder
+    webapp.add_api_controllers( 'galaxy.webapps.galaxy.api', app )
+    # Add the api folder VERSION 2
+    webapp.add_api_controllers( 'galaxy.webapps.galaxy.api.v2', app )
+
     # Force /history to go to /root/history -- needed since the tests assume this
     webapp.add_route( '/history', controller='root', action='history' )
     # Force /activate to go to the controller
     webapp.add_route( '/activate', controller='user', action='activate' )
-    # These two routes handle our simple needs at the moment
+    
     webapp.add_route( '/async/:tool_id/:data_id/:data_secret', controller='async', action='index', tool_id=None, data_id=None, data_secret=None )
     webapp.add_route( '/:controller/:action', action='index' )
     webapp.add_route( '/:action', controller='root', action='index' )
@@ -66,8 +67,8 @@
     webapp.add_route( '/datasets/:dataset_id/display/{filename:.+?}', controller='dataset', action='display', dataset_id=None, filename=None)
     webapp.add_route( '/datasets/:dataset_id/:action/:filename', controller='dataset', action='index', dataset_id=None, filename=None)
     webapp.add_route( '/display_application/:dataset_id/:app_name/:link_name/:user_id/:app_action/:action_param',
-        controller='dataset', action='display_application', dataset_id=None, user_id=None,
-        app_name = None, link_name = None, app_action = None, action_param = None )
+                      controller='dataset', action='display_application', dataset_id=None, user_id=None,
+                      app_name = None, link_name = None, app_action = None, action_param = None )
     webapp.add_route( '/u/:username/d/:slug/:filename', controller='dataset', action='display_by_username_and_slug', filename=None )
     webapp.add_route( '/u/:username/p/:slug', controller='page', action='display_by_username_and_slug' )
     webapp.add_route( '/u/:username/h/:slug', controller='history', action='display_by_username_and_slug' )
@@ -75,22 +76,6 @@
     webapp.add_route( '/u/:username/v/:slug', controller='visualization', action='display_by_username_and_slug' )
     webapp.add_route( '/search', controller='search', action='index' )
 
-    # Add the web API
-    webapp.add_api_controllers( 'galaxy.webapps.galaxy.api', app )
-    # The /folders section is experimental at this point:
-    log.debug( "app.config.api_folders: %s" % app.config.api_folders )
-    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
-    webapp.mapper.resource( 'content', 'contents',
-                                controller='folder_contents',
-                                name_prefix='folder_',
-                                path_prefix='/api/folders/:folder_id',
-                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
-    webapp.mapper.resource( 'content',
-                                'contents',
-                                controller='library_contents',
-                                name_prefix='library_',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
     webapp.mapper.resource( 'content',
                                 'contents',
                                 controller='history_contents',
@@ -102,10 +87,6 @@
                               controller="datasets",
                               action="display",
                               conditions=dict(method=["GET"]))
-    webapp.mapper.resource( 'permission',
-                                'permissions',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
     webapp.mapper.resource( 'user',
                                 'users',
                                 controller='group_users',
@@ -127,11 +108,6 @@
     _add_item_tags_controller( webapp,
                                name_prefix="workflow_",
                                path_prefix='/api/workflows/:workflow_id' )
-
-    _add_item_extended_metadata_controller( webapp,
-                               name_prefix="library_dataset_",
-                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
-
     _add_item_annotation_controller( webapp,
                                name_prefix="history_content_",
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
@@ -147,31 +123,58 @@
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
 
     webapp.mapper.resource( 'dataset', 'datasets', path_prefix='/api' )
-    webapp.mapper.resource_with_deleted( 'library', 'libraries', path_prefix='/api' )
     webapp.mapper.resource( 'sample', 'samples', path_prefix='/api' )
     webapp.mapper.resource( 'request', 'requests', path_prefix='/api' )
     webapp.mapper.resource( 'form', 'forms', path_prefix='/api' )
     webapp.mapper.resource( 'request_type', 'request_types', path_prefix='/api' )
     webapp.mapper.resource( 'role', 'roles', path_prefix='/api' )
     webapp.mapper.resource( 'group', 'groups', path_prefix='/api' )
-    webapp.mapper.resource_with_deleted( 'quota', 'quotas', path_prefix='/api' )
     webapp.mapper.resource( 'tool', 'tools', path_prefix='/api' )
-    webapp.mapper.resource_with_deleted( 'user', 'users', path_prefix='/api' )
     webapp.mapper.resource( 'genome', 'genomes', path_prefix='/api' )
     webapp.mapper.resource( 'visualization', 'visualizations', path_prefix='/api' )
     webapp.mapper.resource( 'workflow', 'workflows', path_prefix='/api' )
-    webapp.mapper.resource_with_deleted( 'history', 'histories', path_prefix='/api' )
     webapp.mapper.resource( 'configuration', 'configuration', path_prefix='/api' )
     #webapp.mapper.connect( 'run_workflow', '/api/workflow/{workflow_id}/library/{library_id}', controller='workflows', action='run', workflow_id=None, library_id=None, conditions=dict(method=["GET"]) )
     webapp.mapper.resource( 'search', 'search', path_prefix='/api' )
 
+    webapp.mapper.resource_with_deleted( 'quota', 'quotas', path_prefix='/api' )
+    webapp.mapper.resource_with_deleted( 'history', 'histories', path_prefix='/api' )
+    webapp.mapper.resource_with_deleted( 'user', 'users', path_prefix='/api' )
+
+    # =======================
+    # ===== LIBRARY API =====
+    # =======================
+
+    webapp.mapper.connect( 'show_lda_item', '/api/libraries/datasets/:id', controller='lda_datasets', action='show', conditions=dict(method=["GET"]) )
+
+    webapp.mapper.resource_with_deleted( 'library', 'libraries', path_prefix='/api' )
+    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
+
+    webapp.mapper.resource( 'content', 'contents',
+                                controller='folder_contents',
+                                name_prefix='folder_',
+                                path_prefix='/api/folders/:folder_id',
+                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
+    webapp.mapper.resource( 'content',
+                                'contents',
+                                controller='library_contents',
+                                name_prefix='library_',
+                                path_prefix='/api/libraries/:library_id',
+                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    webapp.mapper.resource( 'permission',
+                                'permissions',
+                                path_prefix='/api/libraries/:library_id',
+                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    _add_item_extended_metadata_controller( webapp,
+                               name_prefix="library_dataset_",
+                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
+    
+
     # add as a non-ATOM API call to support the notion of a 'current/working' history unique to the history resource
-    webapp.mapper.connect( "set_as_current", "/api/histories/{id}/set_as_current",
-        controller="histories", action="set_as_current", conditions=dict( method=["POST"] ) )
+    webapp.mapper.connect( "set_as_current", "/api/histories/{id}/set_as_current", controller="histories", action="set_as_current", conditions=dict( method=["POST"] ) )
 
     # visualizations registry generic template renderer
-    webapp.add_route( '/visualization/show/:visualization_name',
-        controller='visualization', action='render', visualization_name=None )
+    webapp.add_route( '/visualization/show/:visualization_name', controller='visualization', action='render', visualization_name=None )
 
     # "POST /api/workflows/import"  =>  ``workflows.import_workflow()``.
     # Defines a named route "import_workflow".
@@ -179,6 +182,15 @@
     webapp.mapper.connect("workflow_dict", '/api/workflows/{workflow_id}/download', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
     # Preserve the following download route for now for dependent applications  -- deprecate at some point
     webapp.mapper.connect("workflow_dict", '/api/workflows/download/{workflow_id}', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
+    
+    
+    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
+    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
+                      controller='admin_toolshed',
+                      action='display_image_in_repository',
+                      repository_id=None,
+                      image_file=None )
+
     # Galaxy API for tool shed features.
     webapp.mapper.resource( 'tool_shed_repository',
                             'tool_shed_repositories',
@@ -191,6 +203,7 @@
                             path_prefix='/api',
                             new={ 'install_repository_revision' : 'POST' },
                             parent_resources=dict( member_name='tool_shed_repository', collection_name='tool_shed_repositories' ) )
+
     # Connect logger from app
     if app.trace_logger:
         webapp.trace_logger = app.trace_logger
diff -r 518ded2c3f30cca803a29f76da2e80505b166647 -r 3ea8be4161ab0a9ed2e1c17160c9d3d52d41b555 lib/galaxy/webapps/galaxy/controllers/library.py
--- a/lib/galaxy/webapps/galaxy/controllers/library.py
+++ b/lib/galaxy/webapps/galaxy/controllers/library.py
@@ -80,18 +80,13 @@
     @web.expose
     def list( self, trans, **kwd ):
         params = util.Params( kwd )
-
         # define app configuration for generic mako template
         app = {
             'jscript'       : "galaxy.library"
         }
-
         # fill template
         return trans.fill_template('galaxy.panels.mako', config = {'app' : app})
-    
-    
-#         return trans.fill_template( "/library/list.mako")
-    
+
     @web.expose
     def index( self, trans, **kwd ):
         params = util.Params( kwd )
diff -r 518ded2c3f30cca803a29f76da2e80505b166647 -r 3ea8be4161ab0a9ed2e1c17160c9d3d52d41b555 static/scripts/galaxy.library.js
--- a/static/scripts/galaxy.library.js
+++ b/static/scripts/galaxy.library.js
@@ -7,7 +7,7 @@
 var library_router  = null;
 
 // dependencies
-define(["galaxy.modal", "galaxy.master", "galaxy.library.router"], function(mod_modal, mod_master, router) {
+define(["galaxy.modal", "galaxy.master"], function(mod_modal, mod_master) {
 
 // MMMMMMMMMMMMMMM
 // === Models ====
@@ -15,40 +15,65 @@
 
     // LIBRARY
     var Library = Backbone.Model.extend({
-      urlRoot: '/api/libraries',
-    });
+      urlRoot: '/api/libraries'
+  });
 
     // LIBRARIES
     var Libraries = Backbone.Collection.extend({
       url: '/api/libraries',
       model: Library
-    });
+  });
 
     // ITEM
     var Item = Backbone.Model.extend({
+        urlRoot : '/api/libraries/datasets'
     })
 
     // FOLDER
     var Folder = Backbone.Collection.extend({
       model: Item      
-    })
+  })
 
-    // Container for folder contents (folders, items and metadata).
+    // CONTAINER for folder contents (folders, items and metadata).
     var FolderContainer = Backbone.Model.extend({
-       defaults: {
-        folder: new Folder(),
-        full_path: "unknown",
-        urlRoot: "/api/folders/",
-        id: "unknown"
-      },
-      parse: function(obj) {
-          this.full_path = obj[0].full_path;
-          // this.folder.reset(obj[1].folder_contents);
+     defaults : {
+        folder : new Folder(),
+        full_path : "unknown",
+        urlRoot : "/api/folders/",
+        id : "unknown"
+    },
+    parse : function(obj) {
+      this.full_path = obj[0].full_path;
           // update the inner collection
           this.get("folder").reset(obj[1].folder_contents);
           return obj;
+      }
+  })
+
+    // HISTORY ITEM
+    var HistoryItem = Backbone.Model.extend({
+        urlRoot : '/api/histories/'
+    });
+
+    // HISTORY
+    var GalaxyHistory = Backbone.Model.extend({
+        url : '/api/histories/'
+    });
+
+    // HISTORIES
+    var GalaxyHistories = Backbone.Collection.extend({
+        url : '/api/histories',
+        model : GalaxyHistory
+    });
+
+    //ROUTER
+    var LibraryRouter = Backbone.Router.extend({    
+        routes: {
+            "" :                 "libraries",
+            "folders/:id" :      "folder_content"
         }
-    })
+    });
+
 
 // MMMMMMMMMMMMMM
 // === Views ====
@@ -56,67 +81,155 @@
 
 // galaxy folder
 var FolderContentView = Backbone.View.extend({
-      el : '#center',
+    //main element
+    el : '#center',
     // initialize
     initialize : function(){
-        // view = this;
-        //set up the library router
-        // this.set_up_router();
+        this.folders = [];
+    },    
+    // set up 
+    templateFolder : function (){
+        var tmpl_array = [];
 
-        //render
-        // this.render();
-    },    
-   // set up router
-    set_up_router : function(){   
-        if (library_router === null){
-            library_router = new router.LibraryRouter();
-            Backbone.history.start();   
-        }
+        tmpl_array.push('<a href="#">Libraries</a> | ');
+        tmpl_array.push('<% _.each(path, function(path_item) { %>'); //breadcrumb
+            tmpl_array.push('<% if (path_item[0] != id) { %>');
+            tmpl_array.push('<a href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a> |');
+            tmpl_array.push('<% } else { %>');
+            tmpl_array.push('<%- path_item[1] %>');
+            tmpl_array.push('<% } %>');
+            tmpl_array.push('<% }); %>');
+            tmpl_array.push('<table class="table table-hover table-condensed">');
+            tmpl_array.push('   <thead>');
+            tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
+            tmpl_array.push('       <th>name</th>');
+            tmpl_array.push('       <th>type</th>');
+            tmpl_array.push('   </thead>');
+            tmpl_array.push('   <tbody>');
+            tmpl_array.push('       <% _.each(items, function(content_item) { %>');
+                tmpl_array.push('       <tr class="folder_row" id="<%- content_item.id %>">');
+                tmpl_array.push('       <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
+        tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>'); //folder
+        tmpl_array.push('           <td><a href="#/folders/<%- content_item.id %>"><%- content_item.get("name") %></a>');
+        tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>'); //empty folder
+        tmpl_array.push('           <span class="muted">(empty folder)</span>');
+        tmpl_array.push('           <% } %>');
+        tmpl_array.push('           </td>');
+        tmpl_array.push('           <% } else {  %>');
+        tmpl_array.push('           <td><a class="library-dataset" href="#"><%- content_item.get("name") %></a></td>'); //dataset
+        tmpl_array.push('           <% } %>  ');
+        tmpl_array.push('           <td><%= _.escape(content_item.get("type")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('       ');
+        tmpl_array.push('       ');
+        tmpl_array.push('       ');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+
+        return tmpl_array.join('');
     },
-      template_folder : function (){
-                tmpl_array = [];
+    templateDatasetModal : function(){
+        var tmpl_array = [];
 
-                tmpl_array.push('<% _.each(path, function(path_item) { %>');
-                tmpl_array.push('<% if (path_item[0] != id) { %>');
-                tmpl_array.push('<a href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a> |');
-                tmpl_array.push('<% } else { %>');
-                tmpl_array.push('<%- path_item[1] %>');
-                tmpl_array.push('<% } %>');
-                tmpl_array.push('<% }); %>');
-                tmpl_array.push('<table class="table table-hover table-condensed">');
-                tmpl_array.push('   <thead>');
-                tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
-                tmpl_array.push('       <th>name</th>');
-                tmpl_array.push('       <th>type</th>');
-                tmpl_array.push('   </thead>');
-                tmpl_array.push('   <tbody>');
-                tmpl_array.push('       <% _.each(items, function(content_item) { %>');
-                tmpl_array.push('       <tr class="folder_row" id="<%- content_item.id %>">');
-                tmpl_array.push('           <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
-                tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>');
-                tmpl_array.push('           <td><a href="#/folders/<%- content_item.id %>"><%- content_item.get("name") %></a>');
-                tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>');
-                tmpl_array.push('           <span class="muted">(empty folder)</span>');
-                tmpl_array.push('           <% } %>');
-                tmpl_array.push('           </td>');
-                tmpl_array.push('           <% } else {  %>');
-                tmpl_array.push('           <td><%- content_item.get("name") %></td>');
-                tmpl_array.push('           <% } %>  ');
-                tmpl_array.push('           <td><%= _.escape(content_item.get("type")) %></td>');
-                tmpl_array.push('       </tr>');
-                tmpl_array.push('       <% }); %>');
-                tmpl_array.push('       ');
-                tmpl_array.push('       ');
-                tmpl_array.push('       ');
-                tmpl_array.push('   </tbody>');
-                tmpl_array.push('</table>');
+        tmpl_array.push('<div id="dataset_info_modal">');
+        tmpl_array.push('   <table class="table table-striped table-condensed">');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Name</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("name")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Data type</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("data_type")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Genome build</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("genome_build")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('           <th scope="row">Size</th>');
+        tmpl_array.push('           <td><%= _.escape(size) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Date uploaded</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("date_uploaded")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Uploaded by</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("uploaded_by")) %></td>');
+        tmpl_array.push('       </tr>');
+        // tmpl_array.push('    </table>');
+        // tmpl_array.push('    <hr/>');
+        // tmpl_array.push('    <table class="table table-striped">');
+        tmpl_array.push('           <tr scope="row">');
+        tmpl_array.push('           <th scope="row">Data Lines</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_data_lines")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <th scope="row">Comment Lines</th>');
+        tmpl_array.push('           <% if (item.get("metadata_comment_lines") === "") { %>'); //folder        
+        tmpl_array.push('               <td scope="row"><%= _.escape(item.get("metadata_comment_lines")) %></td>');
+        tmpl_array.push('           <% } else { %>'); 
+        tmpl_array.push('               <td scope="row">unknown</td>');    
+        tmpl_array.push('           <% } %>'); 
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Number of Columns</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_columns")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Column Types</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_column_types")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Miscellaneous information</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("misc_blurb")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('       ');
+        tmpl_array.push('   </table>');
+        // tmpl_array.push('   <hr/>');
+        tmpl_array.push('   <pre class="peek">');
+        tmpl_array.push('   </pre>');
+        tmpl_array.push('</div>');
 
-                return tmpl_array.join('');
-      },
-      events: {
-        'click #select-all-checkboxes' : 'selectAll',
-        'click .folder_row' : 'selectClicked'
-      },
+        return tmpl_array.join('');
+    },
+    templateHistorySelectInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_footer" style="width:60%;">');
+        tmpl_array.push('<select name="history_import" style="width:60%; margin-left: 2em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+            tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+            tmpl_array.push('   <% }); %>');
+            tmpl_array.push('</span>');
+            tmpl_array.push('</div>');
+            tmpl_array.push('');
+
+            return tmpl_array.join('');
+        },
+      // to string
+      size_to_string : function (size)
+      {
+            // identify unit
+            var unit = "";
+            if (size >= 100000000000)   { size = size / 100000000000; unit = "TB"; } else
+            if (size >= 100000000)      { size = size / 100000000; unit = "GB"; } else
+            if (size >= 100000)         { size = size / 100000; unit = "MB"; } else
+            if (size >= 100)            { size = size / 100; unit = "KB"; } else
+            { size = size * 10; unit = "b"; }
+            // return formatted string
+            return (Math.round(size) / 10) + unit;
+        },
+        events: {
+            'click #select-all-checkboxes' : 'selectAll',
+            'click .folder_row' : 'selectClicked',
+            'click .library-dataset' : 'showDatasetDetails'
+        },
+      //self modal
+      modal : null,
+      //loaded folders
+      folders : null,
+      //render the view
       render: function (options) {
         var that = this;
 
@@ -125,80 +238,131 @@
 
         folderContainer.fetch({
           success: function (container) {
-          // folderContainer.attributes.folder = container.attributes.folder;
-          var template = _.template(that.template_folder(), {path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id});
-          that.$el.html(template);
+              // folderContainer.attributes.folder = container.attributes.folder;
+              var template = _.template(that.templateFolder(), {path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id});
+              that.$el.html(template);
           }
-        })
-      },
-      selectAll : function (ev) {
-         var selected = ev.target.checked;
-         // Iterate each checkbox
-         $(':checkbox').each(function () { this.checked = selected; });
-      },
-      selectClicked : function (ev) {
-        var checkbox = $("#" + ev.target.parentElement.id).find(':checkbox')
-        if (checkbox[0] != undefined) {
-          if (checkbox[0].checked){
-            checkbox[0].checked = '';
-          } else {
-            checkbox[0].checked = 'selected';
-          }
+      })
+    },
+      //show modal with dataset info
+      showDatasetDetails : function(e){
+        // prevent default
+        e.preventDefault();
+        
+        //TODO check whether we already have the data
+
+        //load the ID of the row
+        var id = $(e.target).parent().parent().attr('id');
+        
+        //create new item
+        var item = new Item();
+        var histories = new GalaxyHistories();
+        item.id = id;
+        var self = this;
+        //fetch the dataset info
+        item.fetch({
+          success: function (item) {
+                //fetch user histories for import 
+                histories.fetch({
+                    success: function (histories){self.modalFetchSuccess(item, histories)}
+                });
+            }
+        });
+    },
+
+    modalFetchSuccess : function(item, histories){
+        var histories_modal = this.templateHistorySelectInModal();
+        var size = this.size_to_string(item.get('file_size'));
+        var template = _.template(this.templateDatasetModal(), { item : item, size : size });
+            // make modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal({
+                title   : 'Dataset Details',
+                body    : template,
+                buttons : {
+                    'Import' : function() {self.importIntoHistory()},
+                    // 'Notify' : function() {self.modal.showNotification("TEST")},
+                    'Close'  : function() {self.modal.hide()}
+                }
+            });
+            $(".peek").html(item.get("peek"));
+            // this.modal.hideButton('Import');
+            // $(this.modal.elMain).find('.modal-footer').prepend("<div>BUBUBUBU" + "</div>");
+            var history_footer_tmpl = _.template(this.templateHistorySelectInModal(), {histories : histories.models});
+
+            $(this.modal.elMain).find('.buttons').prepend(history_footer_tmpl);
+
+            // show the prepared modal
+            this.modal.show();
+        },
+
+        importIntoHistory : function(){
+            var history_id = 'a0c15f4d91084599';
+            var library_dataset_id = '03501d7626bd192f';
+
+            var historyItem = new HistoryItem();
+            var self = this;
+            historyItem.url = historyItem.urlRoot + history_id + '/contents';
+            console.log(historyItem);
+            historyItem.save({ content : library_dataset_id, source : 'library' }, { success : function(){
+                self.modal.showNotification('Dataset imported', 3000, '#e1f4e0', '#32a427');
+            }
+        });
+        },
+
+        selectAll : function (ev) {
+           var selected = ev.target.checked;
+                 // Iterate each checkbox
+                 $(':checkbox').each(function () { this.checked = selected; });
+             },
+             selectClicked : function (ev) {
+                var checkbox = $("#" + ev.target.parentElement.id).find(':checkbox')
+                if (checkbox[0] != undefined) {
+                  if (checkbox[0].checked){
+                    checkbox[0].checked = '';
+                } else {
+                    checkbox[0].checked = 'selected';
+                }
+            }
         }
-      }
-});
+    });
 
 
 // galaxy library
 var GalaxyLibraryview = Backbone.View.extend({
     el: '#center',
 
-   events: {
+    events: {
         'click #create_new_library_btn' : 'show_library_modal'
-      },
+    },
 
-       
     // initialize
     initialize : function(){
-        // view = this;
-        //set up the libray router
-        // this.set_up_router();
-
-        //render
-        // this.render();
     },    
 
-    // set up router
-    set_up_router : function(){
-        if (library_router === null){
-            library_router = new router.LibraryRouter();
-            Backbone.history.start();   
-        }
-    },
-
     // template
     template_library_list : function (){
         tmpl_array = [];
 
         tmpl_array.push('');
         tmpl_array.push('<h1>Welcome to the data libraries</h1>');
-        tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary">New Library</a>');
+        tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary icon-file ">New Library</a>');
         tmpl_array.push('<table class="table table-striped">');
         tmpl_array.push('   <thead>');
         tmpl_array.push('    <th>name</th>');
         tmpl_array.push('     <th>description</th>');
         tmpl_array.push('     <th>synopsis</th> ');
         tmpl_array.push('     <th>model type</th> ');
-        tmpl_array.push('     <th>id</th> ');
+        // tmpl_array.push('     <th>id</th> ');
         tmpl_array.push('   </thead>');
         tmpl_array.push('   <tbody>');
         tmpl_array.push('       <% _.each(libraries, function(library) { %>');
-        tmpl_array.push('           <tr>');
-        tmpl_array.push('               <td><a href="#/folders/<%- library.id %>"><%- library.get("name") %></a></td>');
-        tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
-        tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
-        tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
-        tmpl_array.push('               <td><a href="#/folders/<%- library.id %>"><%= _.escape(library.get("id")) %></a></td>');
+            tmpl_array.push('           <tr>');
+            tmpl_array.push('               <td><a href="#/folders/<%- library.id %>"><%- library.get("name") %></a></td>');
+            tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
+            tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
+            tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
+        // tmpl_array.push('               <td><a href="#/folders/<%- library.id %>"><%= _.escape(library.get("id")) %></a></td>');
         tmpl_array.push('           </tr>');
         tmpl_array.push('       <% }); %>');
         tmpl_array.push('');
@@ -219,18 +383,17 @@
         // }
         libraries.fetch({
           success: function (libraries) {
-            var template = _.template(that.template_library_list(), {libraries: libraries.models});
+            var template = _.template(that.template_library_list(), { libraries : libraries.models });
             that.$el.html(template);
-          }
-        })
+        }
+    })
     },
 
     // own modal
     modal : null,
 
     // show/hide create library modal
-    show_library_modal : function (e)
-    {
+    show_library_modal : function (e){
         // prevent default
         e.preventDefault();
         
@@ -255,15 +418,21 @@
     create_new_library_event: function(){
         var libraryDetails = this.serialize_new_library();
         var library = new Library();
+        var self = this;
         library.save(libraryDetails, {
           success: function (library) {
-
-            library_router.navigate('', {trigger: true})
-          }
-        });
-        // console.log(libraryDetails);
+            self.modal.hide();
+            self.clear_library_modal();
+            self.render();
+        }
+    });
         return false;
     },
+    clear_library_modal : function(){
+        $("input[name='Name']").val('');
+        $("input[name='Description']").val('');
+        $("input[name='Synopsis']").val('');
+    },
     serialize_new_library : function(){
         return {
             name: $("input[name='Name']").val(),
@@ -279,7 +448,6 @@
         tmpl_array.push('<form>');
         tmpl_array.push('<input type="text" name="Name" value="" placeholder="Name">');
         tmpl_array.push('<input type="text" name="Description" value="" placeholder="Description">');
-        // tmpl_array.push('<label>Synopsis</label>');
         tmpl_array.push('<input type="text" name="Synopsis" value="" placeholder="Synopsis">');
         tmpl_array.push('');
         tmpl_array.push('</form>');
@@ -300,18 +468,18 @@
         folderContentView = new FolderContentView();
         galaxyLibraryview = new GalaxyLibraryview();
 
-        library_router = new router.LibraryRouter();
-            
+        library_router = new LibraryRouter();
+
         library_router.on('route:libraries', function() {
           // render libraries list
           galaxyLibraryview.render();
-        })           
-         
+      })           
+
         library_router.on('route:folder_content', function(id) {
           // render folder's contents
           folderContentView.render({id: id});
-        
-        })      
+
+      })      
 
         // library_router.on('route:show_library_modal', function() {
         //   // render folder's contents
@@ -319,10 +487,10 @@
         
         // })
 
-        Backbone.history.start();   
+Backbone.history.start();   
 
-        return this
-    }
+return this
+}
 });
 
 // return
diff -r 518ded2c3f30cca803a29f76da2e80505b166647 -r 3ea8be4161ab0a9ed2e1c17160c9d3d52d41b555 static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -82,6 +82,7 @@
         this.$footer  = (this.$el).find('.modal-footer');
         this.$buttons = (this.$el).find('.buttons');
         this.$backdrop = (this.$el).find('.modal-backdrop');
+        this.$notification = (this.$el).find('.notification-modal');
         
         // append body
         this.$body.html(options.body);
@@ -122,6 +123,35 @@
         this.$buttons.find('#' + String(name).toLowerCase()).prop('disabled', true);
     },
     
+    // hide buttons
+    hideButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).hide();
+    },
+    // show buttons
+    showButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).show();
+    },
+
+    // show notification
+    showNotification : function(message, duration, bgColor, txtColor) {
+        // defaults
+        var duration = typeof duration !== 'undefined' ? duration : 1500;
+        var bgColor = typeof bgColor !== 'undefined' ? bgColor : "#F4E0E1";
+        var txtColor = typeof txtColor !== 'undefined' ? txtColor : "#A42732";
+
+        var HTMLmessage = "<div class='notification-message' style='text-align:center; line-height:16px; '> " + message + " </div>";
+        this.$notification.html("<div id='notification-bar' style='display:none; float: right; height: 16px; width:100%; background-color: " + bgColor + "; z-index: 100; color: " + txtColor + ";border-bottom: 1px solid " + txtColor + ";'>" + HTMLmessage + "</div>");
+
+        var self = this;
+        /*animate the bar*/
+        $('#notification-bar').slideDown(function() {
+            setTimeout(function() {
+                $('#notification-bar').slideUp(function() {self.$notification.html('');});
+            }, duration);
+        });
+
+    },
+
     // returns scroll top for body element
     scrollTop: function()
     {
@@ -137,10 +167,11 @@
         return  '<div class="modal">' +
                     '<div class="modal-backdrop fade in" style="z-index: -1;"></div>' +
                     '<div class="modal-dialog">' +
-                        '<div class="modal-content">' +
+                        '<div class="modal-content"">' +
                             '<div class="modal-header">' +
                                 '<button type="button" class="close" style="display: none;">×</button>' +
                                 '<h4 class="title">' + title + '</h4>' +
+                                '<span class="notification-modal"></span>' + 
                             '</div>' +
                             '<div class="modal-body"></div>' +
                             '<div class="modal-footer">' +
https://bitbucket.org/galaxy/galaxy-central/commits/2bb7df095a5a/
Changeset:   2bb7df095a5a
Branch:      data_library
User:        martenson
Date:        2013-12-03 22:43:06
Summary:     slight refactoring and bugfix of db_next_hid generator
Affected #:  1 file
diff -r 3ea8be4161ab0a9ed2e1c17160c9d3d52d41b555 -r 2bb7df095a5a23e3dce2e02fb963a30ff3b7d377 lib/galaxy/model/mapping.py
--- a/lib/galaxy/model/mapping.py
+++ b/lib/galaxy/model/mapping.py
@@ -1988,8 +1988,18 @@
     table = self.table
     trans = conn.begin()
     try:
-        next_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
-        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid + 1 ) )
+        
+#         log.debug("XXXXXXXXXXXXXXX NEW GENERATOOOOOR")
+#         next_hid = table.update() \
+#         .returning( table.c.hid_counter ) \
+#         .where( table.c.id == self.id ) \
+#         .values( hid_counter = hid_counter + 1 )
+#         log.debug('XXXXXXXXXXXXXXXXXXXXXXXXXXX NEXT HID: ' + str(next_hid))
+        
+    
+        current_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
+        next_hid = current_hid + 1
+        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid ) )
         trans.commit()
         return next_hid
     except:
https://bitbucket.org/galaxy/galaxy-central/commits/4e53d7b759d1/
Changeset:   4e53d7b759d1
Branch:      data_library
User:        martenson
Date:        2013-12-03 22:43:25
Summary:     added dropdown menu plugin to bootstrap
Affected #:  1 file
diff -r 2bb7df095a5a23e3dce2e02fb963a30ff3b7d377 -r 4e53d7b759d1999758a30611e5d897e3e1d20782 static/scripts/libs/bootstrap.js
--- a/static/scripts/libs/bootstrap.js
+++ b/static/scripts/libs/bootstrap.js
@@ -575,3 +575,158 @@
   }
 
 }(window.jQuery);
+
+/* ========================================================================
+ * Bootstrap: dropdown.js v3.0.0
+ * http://twbs.github.com/bootstrap/javascript.html#dropdowns
+ * ========================================================================
+ * Copyright 2012 Twitter, Inc.
+ *
+ * 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.
+ * ======================================================================== */
+
+
++function ($) { "use strict";
+
+  // DROPDOWN CLASS DEFINITION
+  // =========================
+
+  var backdrop = '.dropdown-backdrop'
+  var toggle   = '[data-toggle=dropdown]'
+  var Dropdown = function (element) {
+    var $el = $(element).on('click.bs.dropdown', this.toggle)
+  }
+
+  Dropdown.prototype.toggle = function (e) {
+    var $this = $(this)
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    clearMenus()
+
+    if (!isActive) {
+      if ('ontouchstart' in document.documentElement && !$parent.closest('.navbar-nav').length) {
+        // if mobile we we use a backdrop because click events don't delegate
+        $('<div class="dropdown-backdrop"/>').insertAfter($(this)).on('click', clearMenus)
+      }
+
+      $parent.trigger(e = $.Event('show.bs.dropdown'))
+
+      if (e.isDefaultPrevented()) return
+
+      $parent
+        .toggleClass('open')
+        .trigger('shown.bs.dropdown')
+
+      $this.focus()
+    }
+
+    return false
+  }
+
+  Dropdown.prototype.keydown = function (e) {
+    if (!/(38|40|27)/.test(e.keyCode)) return
+
+    var $this = $(this)
+
+    e.preventDefault()
+    e.stopPropagation()
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    if (!isActive || (isActive && e.keyCode == 27)) {
+      if (e.which == 27) $parent.find(toggle).focus()
+      return $this.click()
+    }
+
+    var $items = $('[role=menu] li:not(.divider):visible a', $parent)
+
+    if (!$items.length) return
+
+    var index = $items.index($items.filter(':focus'))
+
+    if (e.keyCode == 38 && index > 0)                 index--                        // up
+    if (e.keyCode == 40 && index < $items.length - 1) index++                        // down
+    if (!~index)                                      index=0
+
+    $items.eq(index).focus()
+  }
+
+  function clearMenus() {
+    $(backdrop).remove()
+    $(toggle).each(function (e) {
+      var $parent = getParent($(this))
+      if (!$parent.hasClass('open')) return
+      $parent.trigger(e = $.Event('hide.bs.dropdown'))
+      if (e.isDefaultPrevented()) return
+      $parent.removeClass('open').trigger('hidden.bs.dropdown')
+    })
+  }
+
+  function getParent($this) {
+    var selector = $this.attr('data-target')
+
+    if (!selector) {
+      selector = $this.attr('href')
+      selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
+    }
+
+    var $parent = selector && $(selector)
+
+    return $parent && $parent.length ? $parent : $this.parent()
+  }
+
+
+  // DROPDOWN PLUGIN DEFINITION
+  // ==========================
+
+  var old = $.fn.dropdown
+
+  $.fn.dropdown = function (option) {
+    return this.each(function () {
+      var $this = $(this)
+      var data  = $this.data('dropdown')
+
+      if (!data) $this.data('dropdown', (data = new Dropdown(this)))
+      if (typeof option == 'string') data[option].call($this)
+    })
+  }
+
+  $.fn.dropdown.Constructor = Dropdown
+
+
+  // DROPDOWN NO CONFLICT
+  // ====================
+
+  $.fn.dropdown.noConflict = function () {
+    $.fn.dropdown = old
+    return this
+  }
+
+
+  // APPLY TO STANDARD DROPDOWN ELEMENTS
+  // ===================================
+
+  $(document)
+    .on('click.bs.dropdown.data-api', clearMenus)
+    .on('click.bs.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
+    .on('click.bs.dropdown.data-api'  , toggle, Dropdown.prototype.toggle)
+    .on('keydown.bs.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown)
+
+}(window.jQuery);
\ No newline at end of file
https://bitbucket.org/galaxy/galaxy-central/commits/44399512777b/
Changeset:   44399512777b
Branch:      data_library
User:        martenson
Date:        2013-12-03 22:44:22
Summary:     library work, galaxy modal changes, api v2 preparation, API tweaks, API for dataset downloading implementation
Affected #:  9 files
diff -r 4e53d7b759d1999758a30611e5d897e3e1d20782 -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 lib/galaxy/model/__init__.py
--- a/lib/galaxy/model/__init__.py
+++ b/lib/galaxy/model/__init__.py
@@ -1768,7 +1768,7 @@
 class Library( object, Dictifiable ):
     permitted_actions = get_permitted_actions( filter='LIBRARY' )
     dict_collection_visible_keys = ( 'id', 'name' )
-    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis' )
+    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis', 'root_folder_id' )
     def __init__( self, name=None, description=None, synopsis=None, root_folder=None ):
         self.name = name or "Unnamed library"
         self.description = description
diff -r 4e53d7b759d1999758a30611e5d897e3e1d20782 -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 lib/galaxy/webapps/galaxy/api/datasets.py
--- a/lib/galaxy/webapps/galaxy/api/datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/datasets.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a dataset.
+API operations on the contents of a history dataset.
 """
 from galaxy import web
 from galaxy.visualization.data_providers.genome import FeatureLocationIndexDataProvider
diff -r 4e53d7b759d1999758a30611e5d897e3e1d20782 -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -42,19 +42,34 @@
         GET /api/folders/{encoded_folder_id}/contents
         Displays a collection (list) of a folder's contents (files and folders).
         Encoded folder ID is prepended with 'F' if it is a folder as opposed to a data set which does not have it.
+        Full path is provided as a separate object in response providing data for breadcrumb path building.
         """
         folder_container = []
         current_user_roles = trans.get_current_user_roles()
+        
+#         log.debug( 'XXXXXXXXXXXXXXXXXXXXX GOT id: ' + folder_id )
+#         log.debug( 'XXXXXXXXXXXXXXXXXXXXX decode id: ' + folder_id[1:] )
+#         log.debug( 'XXXXXXXXXXXXXXXXXXXXX call decode: ' + str(trans.security.decode_id( folder_id[1:] )) )
 
-        try:
-            decoded_folder_id = trans.security.decode_id( folder_id[-16:] )
-        except TypeError:
-            trans.response.status = 400
-            return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
+#         try:
+#             decoded_folder_id = trans.security.decode_id( folder_id[-16:] )
+#         except TypeError:
+#             trans.response.status = 400
+#             return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
+#         log.debug( 'XXXXXXXXXXXXXXXXXXXXX decoded id: ' + decoded_folder_id )
+
+        if ( folder_id.startswith( 'F' ) ):
+            try:
+                decoded_folder_id = trans.security.decode_id( folder_id[1:] )
+            except TypeError:
+                trans.response.status = 400
+                return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
 
         try:
             folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( decoded_folder_id )
-            parent_library = folder.parent_library
+#             log.debug( 'XXXXXXXXXXXXXXXXXXXXX FOLDER id: ' + str(folder.id) )
+#             log.debug( 'XXXXXXXXXXXXXXXXXXXXX FOLDER name: ' + str(folder.name) )
+#             parent_library = folder.parent_library
         except:
             folder = None
             log.error( "FolderContentsController.index: Unable to retrieve folder with ID: %s" % folder_id )
@@ -77,13 +92,17 @@
             path_to_root = []
             # We are almost in root
             if folder.parent_id is None:
-                log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ALMOST ROOT FOLDER! ADDING: " + str( folder.name ) )
+                
+#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ALMOST ROOT FOLDER! ADDING ID: " + str( folder.id ) )
+#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ALMOST ROOT FOLDER! ADDING NAME: " + str( folder.name ) )
                 path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
 #                 upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_library.id )
 #                 path_to_root.append( ( upper_folder.id, upper_folder.name ) )
             else:
             # We add the current folder and traverse up one folder.
-                log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ADDING THIS FOLDER AND TRAVERSING UP:  " + str( folder.name ) )
+#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX folder.parent_id " + str( folder.parent_id ) )
+#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ALMOST ROOT FOLDER! ADDING ID: " + str( folder.id ) )
+#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ADDING THIS FOLDER AND TRAVERSING UP:  " + str( folder.name ) )
                 path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
                 upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_id )
                 path_to_root.extend( build_path( upper_folder ) )
diff -r 4e53d7b759d1999758a30611e5d897e3e1d20782 -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- a/lib/galaxy/webapps/galaxy/api/lda_datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -1,27 +1,69 @@
 """
-API operations on the contents of a dataset from library.
+API operations on the dataset from library.
 """
-from galaxy import web
+import glob
+import logging
+import operator
+import os
+import os.path
+import string
+import sys
+import tarfile
+import tempfile
+import urllib
+import urllib2
+import zipfile
+from galaxy.security import Action
+from galaxy import util, web
+from galaxy.util.streamball import StreamBall
 from galaxy.web.base.controller import BaseAPIController, UsesLibraryMixinItems
 
 import logging
 log = logging.getLogger( __name__ )
 
+# Test for available compression types
+# tmpd = tempfile.mkdtemp()
+# comptypes = []
+# for comptype in ( 'gz', 'bz2' ):
+#     tmpf = os.path.join( tmpd, 'compression_test.tar.' + comptype )
+#     try:
+#         archive = tarfile.open( tmpf, 'w:' + comptype )
+#         archive.close()
+#         comptypes.append( comptype )
+#     except tarfile.CompressionError:
+#         log.exception( "Compression error when testing %s compression.  This option will be disabled for library downloads." % comptype )
+#     try:
+#         os.unlink( tmpf )
+#     except OSError:
+#         pass
+ziptype = '32'
+# tmpf = os.path.join( tmpd, 'compression_test.zip' )
+# try:
+#     archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
+#     archive.close()
+#     comptypes.append( 'zip' )
+#     ziptype = '64'
+# except RuntimeError:
+#     log.exception( "Compression error when testing zip compression. This option will be disabled for library downloads." )
+# except (TypeError, zipfile.LargeZipFile):
+#     # ZIP64 is only in Python2.5+.  Remove TypeError when 2.4 support is dropped
+#     log.warning( 'Max zip file size is 2GB, ZIP64 not supported' )
+#     comptypes.append( 'zip' )
+# try:
+#     os.unlink( tmpf )
+# except OSError:
+#     pass
+# os.rmdir( tmpd )
+
+
+
 class DatasetsController( BaseAPIController, UsesLibraryMixinItems ):
 
     @web.expose_api
-    def index( self, trans, **kwd ):
-        """
-        GET /api/libraries/datasets
-        """
-        trans.response.status = 501
-        return 'not implemented'
-
-    @web.expose_api
     def show( self, trans, id, **kwd ):
         """
         GET /api/libraries/datasets/{encoded_dataset_id}
-        Displays information about and/or content of a dataset identified by the lda ID.
+        Displays information about the dataset identified by the lda ID.
         """
         # Get dataset.
         try:
@@ -35,4 +77,165 @@
             rval = "Error in dataset API at listing contents: " + str( e )
             log.error( rval + ": %s" % str(e), exc_info=True )
             trans.response.status = 500
-        return rval
\ No newline at end of file
+        
+        rval['id'] = trans.security.encode_id(rval['id']);
+        rval['ldda_id'] = trans.security.encode_id(rval['ldda_id']);
+        rval['folder_id'] = 'f' + trans.security.encode_id(rval['folder_id'])
+        
+        return rval
+    
+    @web.expose
+    def download( self, trans, format, **kwd ):
+        """
+        POST /api/libraries/datasets/download/{format}
+        POST data: ldda_ids = []
+        Downloads dataset(s) in the requested format.
+        """
+        lddas = []
+#         is_admin = trans.user_is_admin()
+#         current_user_roles = trans.get_current_user_roles()
+        
+        datasets_to_download = kwd['ldda_ids%5B%5D']
+        
+        if ( datasets_to_download != None ):
+            datasets_to_download = util.listify( datasets_to_download )
+            for dataset_id in datasets_to_download:
+                try:
+                    ldda = trans.sa_session.query( trans.app.model.LibraryDatasetDatasetAssociation ).get( trans.security.decode_id( dataset_id ) )
+                    assert not ldda.dataset.purged
+                    lddas.append( ldda )
+                except:
+                    ldda = None
+                    message += "Invalid library dataset id (%s) specified.  " % str( dataset_id )
+                    
+        if format in [ 'zip','tgz','tbz' ]:
+                error = False
+                killme = string.punctuation + string.whitespace
+                trantab = string.maketrans(killme,'_'*len(killme))
+                try:
+                    outext = 'zip'
+                    if format == 'zip':
+                        # Can't use mkstemp - the file must not exist first
+                        tmpd = tempfile.mkdtemp()
+                        util.umask_fix_perms( tmpd, trans.app.config.umask, 0777, self.app.config.gid )
+                        tmpf = os.path.join( tmpd, 'library_download.' + format )
+                        if ziptype == '64' and trans.app.config.upstream_gzip:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED, True )
+                        elif ziptype == '64':
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
+                        elif trans.app.config.upstream_gzip:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED )
+                        else:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED )
+                        archive.add = lambda x, y: archive.write( x, y.encode('CP437') )
+                    elif format == 'tgz':
+                        if trans.app.config.upstream_gzip:
+                            archive = StreamBall( 'w|' )
+                            outext = 'tar'
+                        else:
+                            archive = StreamBall( 'w|gz' )
+                            outext = 'tgz'
+                    elif format == 'tbz':
+                        archive = StreamBall( 'w|bz2' )
+                        outext = 'tbz2'
+                except ( OSError, zipfile.BadZipfile ):
+                    error = True
+                    log.exception( "Unable to create archive for download" )
+                    message = "Unable to create archive for download, please report this error"
+                    status = 'error'
+                except:
+                     error = True
+                     log.exception( "Unexpected error %s in create archive for download" % sys.exc_info()[0] )
+                     message = "Unable to create archive for download, please report - %s" % sys.exc_info()[0]
+                     status = 'error'
+                if not error:
+                    composite_extensions = trans.app.datatypes_registry.get_composite_extensions()
+                    seen = []
+                    for ldda in lddas:
+                        if ldda.dataset.state in [ 'new', 'upload', 'queued', 'running', 'empty', 'discarded' ]:
+                            continue
+                        ext = ldda.extension
+                        is_composite = ext in composite_extensions
+                        path = ""
+                        parent_folder = ldda.library_dataset.folder
+                        while parent_folder is not None:
+                            # Exclude the now-hidden "root folder"
+                            if parent_folder.parent is None:
+                                path = os.path.join( parent_folder.library_root[0].name, path )
+                                break
+                            path = os.path.join( parent_folder.name, path )
+                            parent_folder = parent_folder.parent
+                        path += ldda.name
+                        while path in seen:
+                            path += '_'
+                        seen.append( path )
+                        zpath = os.path.split(path)[-1] # comes as base_name/fname
+                        outfname,zpathext = os.path.splitext(zpath)
+                        if is_composite:
+                            # need to add all the components from the extra_files_path to the zip
+                            if zpathext == '':
+                                zpath = '%s.html' % zpath # fake the real nature of the html file
+                            try:
+                                archive.add(ldda.dataset.file_name,zpath) # add the primary of a composite set
+                            except IOError:
+                                error = True
+                                log.exception( "Unable to add composite parent %s to temporary library download archive" % ldda.dataset.file_name)
+                                message = "Unable to create archive for download, please report this error"
+                                status = 'error'
+                                continue
+                            flist = glob.glob(os.path.join(ldda.dataset.extra_files_path,'*.*')) # glob returns full paths
+                            for fpath in flist:
+                                efp,fname = os.path.split(fpath)
+                                if fname > '':
+                                    fname = fname.translate(trantab)
+                                try:
+                                    archive.add( fpath,fname )
+                                except IOError:
+                                    error = True
+                                    log.exception( "Unable to add %s to temporary library download archive %s" % (fname,outfname))
+                                    message = "Unable to create archive for download, please report this error"
+                                    status = 'error'
+                                    continue
+                        else: # simple case
+                            try:
+                                archive.add( ldda.dataset.file_name, path )
+                            except IOError:
+                                error = True
+                                log.exception( "Unable to write %s to temporary library download archive" % ldda.dataset.file_name)
+                                message = "Unable to create archive for download, please report this error"
+                                status = 'error'
+                    if not error:
+                        lname = 'selected_dataset'
+                        fname = lname.replace( ' ', '_' ) + '_files'
+                        if format == 'zip':
+                            archive.close()
+                            trans.response.set_content_type( "application/octet-stream" )
+                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                            archive = util.streamball.ZipBall(tmpf, tmpd)
+                            archive.wsgi_status = trans.response.wsgi_status()
+                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            return archive.stream
+                        else:
+                            trans.response.set_content_type( "application/x-tar" )
+                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                            archive.wsgi_status = trans.response.wsgi_status()
+                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            return archive.stream
+        elif format == 'uncompressed':
+            if len(lddas) != 1:
+                return 'Wrong request'
+            else:
+                single_dataset = lddas[0]
+                trans.response.set_content_type( single_dataset.get_mime() )
+                fStat = os.stat( ldda.file_name )
+                trans.response.headers[ 'Content-Length' ] = int( fStat.st_size )
+                valid_chars = '.,^_-()[]0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
+                fname = ldda.name
+                fname = ''.join( c in valid_chars and c or '_' for c in fname )[ 0:150 ]
+                trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s"' % fname
+                try:
+                    return open( single_dataset.file_name )
+                except:
+                    return 'This dataset contains no content'
+        else:
+            return 'Wrong format';
\ No newline at end of file
diff -r 4e53d7b759d1999758a30611e5d897e3e1d20782 -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 lib/galaxy/webapps/galaxy/api/libraries.py
--- a/lib/galaxy/webapps/galaxy/api/libraries.py
+++ b/lib/galaxy/webapps/galaxy/api/libraries.py
@@ -52,6 +52,7 @@
             item = library.to_dict( view='element' )
             item['url'] = url_for( route, id=trans.security.encode_id( library.id ) )
             item['id'] = 'F' + trans.security.encode_id( item['id'] )
+            item['root_folder_id'] = 'F' + trans.security.encode_id( item['root_folder_id'] )
             rval.append( item )
         return rval
 
diff -r 4e53d7b759d1999758a30611e5d897e3e1d20782 -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -146,6 +146,7 @@
     # =======================
 
     webapp.mapper.connect( 'show_lda_item', '/api/libraries/datasets/:id', controller='lda_datasets', action='show', conditions=dict(method=["GET"]) )
+    webapp.mapper.connect( 'download_lda_items', '/api/libraries/datasets/download/:format', controller='lda_datasets', action='download', conditions=dict(method=["POST"]) )
 
     webapp.mapper.resource_with_deleted( 'library', 'libraries', path_prefix='/api' )
     webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
diff -r 4e53d7b759d1999758a30611e5d897e3e1d20782 -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 static/scripts/galaxy.library.js
--- a/static/scripts/galaxy.library.js
+++ b/static/scripts/galaxy.library.js
@@ -5,6 +5,9 @@
 // global variables
 var view            = null;
 var library_router  = null;
+var responses = [];
+var KEYCODE_ENTER = 13;
+var KEYCODE_ESC = 27;
 
 // dependencies
 define(["galaxy.modal", "galaxy.master"], function(mod_modal, mod_master) {
@@ -70,52 +73,99 @@
     var LibraryRouter = Backbone.Router.extend({    
         routes: {
             "" :                 "libraries",
-            "folders/:id" :      "folder_content"
+            "folders/:id" :      "folder_content",
+            "folders/:folder_id/download/:format" : "download"
         }
     });
 
 
 // MMMMMMMMMMMMMM
-// === Views ====
+// === VIEWS ====
 // MMMMMMMMMMMMMM
 
 // galaxy folder
 var FolderContentView = Backbone.View.extend({
-    //main element
+    // main element definition
     el : '#center',
+    // progress percentage
+    progress: 0,
+    // progress rate per one item
+    progressStep: 1, 
+    // last selected history in modal for UX
+    lastSelectedHistory: '',
+    // self modal
+    modal : null,
+    // loaded folders
+    folders : null,
+    
     // initialize
     initialize : function(){
         this.folders = [];
+        this.queue = jQuery.Deferred();
+        this.queue.resolve();
     },    
+
+// MMMMMMMMMMMMMMMMMM
+// === TEMPLATES ====
+// MMMMMMMMMMMMMMMMMM
+
     // set up 
     templateFolder : function (){
         var tmpl_array = [];
 
-        tmpl_array.push('<a href="#">Libraries</a> | ');
+        // CONTAINER
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; ">');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to Martin.</h3>');
+
+        // TOOLBAR
+        tmpl_array.push('<div id="library_folder_toolbar" >');
+        tmpl_array.push('   <button id="toolbtn_create_folder" class="btn btn-primary" type="button">new folder</button>');
+        tmpl_array.push('   <button id="toolbtn_bulk_import" style="display: none; margin-left: 0.5em;" type="button">Import into History</button>');
+        
+        tmpl_array.push('   <div id="toolbtn_dl" class="btn-group" style="margin-left: 0.5em; display: none; ">');
+        tmpl_array.push('       <button id="drop_toggle" type="button" class="btn btn-default dropdown-toggle" data-toggle="dropdown">');
+        tmpl_array.push('       Download <span class="caret"></span>');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       <ul class="dropdown-menu" role="menu">');
+        // tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/uncompressed">uncompressed</a></li>');
+        // tmpl_array.push('          <li class="divider"></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tgz">.tar.gz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tbz">.tar.bz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/zip">.zip</a></li>');
+        tmpl_array.push('       </ul>');
+        tmpl_array.push('   </div>');
+
+        tmpl_array.push('</div>');
+
+        // BREADCRUMBS
+        tmpl_array.push('<a href="#">Libraries</a><b>|</b> ');
         tmpl_array.push('<% _.each(path, function(path_item) { %>'); //breadcrumb
-            tmpl_array.push('<% if (path_item[0] != id) { %>');
-            tmpl_array.push('<a href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a> |');
-            tmpl_array.push('<% } else { %>');
-            tmpl_array.push('<%- path_item[1] %>');
-            tmpl_array.push('<% } %>');
-            tmpl_array.push('<% }); %>');
-            tmpl_array.push('<table class="table table-hover table-condensed">');
-            tmpl_array.push('   <thead>');
-            tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
-            tmpl_array.push('       <th>name</th>');
-            tmpl_array.push('       <th>type</th>');
-            tmpl_array.push('   </thead>');
-            tmpl_array.push('   <tbody>');
-            tmpl_array.push('       <% _.each(items, function(content_item) { %>');
-                tmpl_array.push('       <tr class="folder_row" id="<%- content_item.id %>">');
-                tmpl_array.push('       <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
+        tmpl_array.push('<% if (path_item[0] != id) { %>');
+        tmpl_array.push('<a href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a><b>|</b> ');
+        tmpl_array.push('<% } else { %>');
+        tmpl_array.push('<%- path_item[1] %>');
+        tmpl_array.push('<% } %>');
+        tmpl_array.push('<% }); %>');
+
+        // FODLER CONTENT
+        tmpl_array.push('<table id="folder_table" class="table table-hover table-condensed">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
+        tmpl_array.push('       <th>name</th>');
+        tmpl_array.push('       <th>type</th>');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <% _.each(items, function(content_item) { %>');
+        tmpl_array.push('       <tr class="folder_row" id="<%- content_item.id %>">');
         tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>'); //folder
-        tmpl_array.push('           <td><a href="#/folders/<%- content_item.id %>"><%- content_item.get("name") %></a>');
+        tmpl_array.push('               <td style="text-align: center; "></td>');
+        tmpl_array.push('               <td><a href="#/folders/<%- content_item.id %>"><%- content_item.get("name") %></a>');
         tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>'); //empty folder
         tmpl_array.push('           <span class="muted">(empty folder)</span>');
         tmpl_array.push('           <% } %>');
         tmpl_array.push('           </td>');
         tmpl_array.push('           <% } else {  %>');
+        tmpl_array.push('           <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
         tmpl_array.push('           <td><a class="library-dataset" href="#"><%- content_item.get("name") %></a></td>'); //dataset
         tmpl_array.push('           <% } %>  ');
         tmpl_array.push('           <td><%= _.escape(content_item.get("type")) %></td>');
@@ -123,10 +173,10 @@
         tmpl_array.push('       <% }); %>');
         tmpl_array.push('       ');
         tmpl_array.push('       ');
-        tmpl_array.push('       ');
         tmpl_array.push('   </tbody>');
         tmpl_array.push('</table>');
 
+        tmpl_array.push('</div>');
         return tmpl_array.join('');
     },
     templateDatasetModal : function(){
@@ -135,7 +185,7 @@
         tmpl_array.push('<div id="dataset_info_modal">');
         tmpl_array.push('   <table class="table table-striped table-condensed">');
         tmpl_array.push('       <tr>');
-        tmpl_array.push('           <th scope="row">Name</th>');
+        tmpl_array.push('           <th scope="row" id="id_row" data-id="<%= _.escape(item.get("ldda_id")) %>">Name</th>');
         tmpl_array.push('           <td><%= _.escape(item.get("name")) %></td>');
         tmpl_array.push('       </tr>');
         tmpl_array.push('       <tr>');
@@ -157,9 +207,6 @@
         tmpl_array.push('           <th scope="row">Uploaded by</th>');
         tmpl_array.push('           <td><%= _.escape(item.get("uploaded_by")) %></td>');
         tmpl_array.push('       </tr>');
-        // tmpl_array.push('    </table>');
-        // tmpl_array.push('    <hr/>');
-        // tmpl_array.push('    <table class="table table-striped">');
         tmpl_array.push('           <tr scope="row">');
         tmpl_array.push('           <th scope="row">Data Lines</th>');
         tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_data_lines")) %></td>');
@@ -183,31 +230,45 @@
         tmpl_array.push('           <th scope="row">Miscellaneous information</th>');
         tmpl_array.push('           <td scope="row"><%= _.escape(item.get("misc_blurb")) %></td>');
         tmpl_array.push('       </tr>');
-        tmpl_array.push('       <tr>');
-        tmpl_array.push('       ');
         tmpl_array.push('   </table>');
-        // tmpl_array.push('   <hr/>');
         tmpl_array.push('   <pre class="peek">');
         tmpl_array.push('   </pre>');
         tmpl_array.push('</div>');
 
         return tmpl_array.join('');
     },
+
     templateHistorySelectInModal : function(){
         var tmpl_array = [];
 
-        tmpl_array.push('<span id="history_modal_footer" style="width:60%;">');
-        tmpl_array.push('<select name="history_import" style="width:60%; margin-left: 2em; "> ');
+        tmpl_array.push('<span id="history_modal_combo" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_single" name="dataset_import_single" style="width:50%; margin-bottom: 1em; "> ');
         tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
-            tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
-            tmpl_array.push('   <% }); %>');
-            tmpl_array.push('</span>');
-            tmpl_array.push('</div>');
-            tmpl_array.push('');
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
 
-            return tmpl_array.join('');
-        },
-      // to string
+        return tmpl_array.join('');
+        },   
+
+    templateBulkImportInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_combo_bulk" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_bulk" name="dataset_import_bulk" style="width:50%; margin-bottom: 1em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
+
+        return tmpl_array.join('');
+        },    
+
+      // convert size to nice string
       size_to_string : function (size)
       {
             // identify unit
@@ -220,17 +281,26 @@
             // return formatted string
             return (Math.round(size) / 10) + unit;
         },
-        events: {
+
+// MMMMMMMMMMMMMMM
+// === EVENTS ====
+// MMMMMMMMMMMMMMM
+
+      // event binding
+      events: {
             'click #select-all-checkboxes' : 'selectAll',
             'click .folder_row' : 'selectClicked',
+            'click #toolbtn_bulk_import' : 'modalBulkImport',
+            'click #toolbtn_dl' : 'bulkDownload',
             'click .library-dataset' : 'showDatasetDetails'
         },
-      //self modal
-      modal : null,
-      //loaded folders
-      folders : null,
-      //render the view
+
+      //render the folder view
       render: function (options) {
+        //hack to show scrollbars
+        $("#center").css('overflow','auto');
+
+        view = this;
         var that = this;
 
         var folderContainer = new FolderContainer({id: options.id});
@@ -241,15 +311,16 @@
               // folderContainer.attributes.folder = container.attributes.folder;
               var template = _.template(that.templateFolder(), {path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id});
               that.$el.html(template);
-          }
-      })
-    },
-      //show modal with dataset info
+            }
+          })
+        },
+
+      //show modal with current dataset info
       showDatasetDetails : function(e){
         // prevent default
         e.preventDefault();
         
-        //TODO check whether we already have the data
+//TODO check whether we already have the data
 
         //load the ID of the row
         var id = $(e.target).parent().parent().attr('id');
@@ -259,53 +330,90 @@
         var histories = new GalaxyHistories();
         item.id = id;
         var self = this;
+
         //fetch the dataset info
         item.fetch({
           success: function (item) {
-                //fetch user histories for import 
+// TODO can render here already
+                //fetch user histories for import purposes
                 histories.fetch({
-                    success: function (histories){self.modalFetchSuccess(item, histories)}
+                    success: function (histories){self.renderModalAfterFetch(item, histories)}
                 });
             }
         });
     },
 
-    modalFetchSuccess : function(item, histories){
-        var histories_modal = this.templateHistorySelectInModal();
+      // show the current dataset in a modal
+      renderModalAfterFetch : function(item, histories){
         var size = this.size_to_string(item.get('file_size'));
         var template = _.template(this.templateDatasetModal(), { item : item, size : size });
+        this.modal = null;
             // make modal
             var self = this;
             this.modal = new mod_modal.GalaxyModal({
                 title   : 'Dataset Details',
                 body    : template,
                 buttons : {
-                    'Import' : function() {self.importIntoHistory()},
-                    // 'Notify' : function() {self.modal.showNotification("TEST")},
-                    'Close'  : function() {self.modal.hide()}
+                    'Import' : function() { self.importCurrentIntoHistory() },
+                    'Download' : function() { self.downloadCurrent() },
+                    'Close'  : function() { self.modal.hide(); $('.modal').html(''); self.modal = null; } // TODO refill nicely modal with data
                 }
             });
             $(".peek").html(item.get("peek"));
-            // this.modal.hideButton('Import');
-            // $(this.modal.elMain).find('.modal-footer').prepend("<div>BUBUBUBU" + "</div>");
             var history_footer_tmpl = _.template(this.templateHistorySelectInModal(), {histories : histories.models});
+            $(this.modal.elMain).find('.buttons').prepend(history_footer_tmpl);
 
-            $(this.modal.elMain).find('.buttons').prepend(history_footer_tmpl);
+            // preset last selected history if we know it
+            if (self.lastSelectedHistory.length > 0) {
+                $(this.modal.elMain).find('#dataset_import_single').val(self.lastSelectedHistory);
+            }
 
             // show the prepared modal
             this.modal.show();
         },
 
-        importIntoHistory : function(){
-            var history_id = 'a0c15f4d91084599';
-            var library_dataset_id = '03501d7626bd192f';
+        // download dataset shown currently in modal
+        downloadCurrent : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
 
+            var library_dataset_id = [];
+            library_dataset_id.push($('#id_row').attr('data-id'));
+            var url = '/api/libraries/datasets/download/uncompressed';
+            var data = {'ldda_ids' : library_dataset_id};
+
+            // we assume the view is existent
+            folderContentView.processDownload(url, data);
+            this.modal.enableButton('Import');
+            this.modal.enableButton('Download');
+        },
+
+        // import dataset shown currently in modal into selected history
+        importCurrentIntoHistory : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
+
+            var history_id = $(this.modal.elMain).find('select[name=dataset_import_single] option:selected').val();
+            this.lastSelectedHistory = history_id; //save selected history for further use
+
+            var library_dataset_id = $('#id_row').attr('data-id');
             var historyItem = new HistoryItem();
             var self = this;
             historyItem.url = historyItem.urlRoot + history_id + '/contents';
-            console.log(historyItem);
+
+            // save the dataset into selected history
             historyItem.save({ content : library_dataset_id, source : 'library' }, { success : function(){
                 self.modal.showNotification('Dataset imported', 3000, '#e1f4e0', '#32a427');
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');
+            }, error : function(){
+                self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, '#f4e0e1', '#a42732');
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');                
             }
         });
         },
@@ -314,9 +422,11 @@
            var selected = ev.target.checked;
                  // Iterate each checkbox
                  $(':checkbox').each(function () { this.checked = selected; });
+                 this.showTools();
              },
-             selectClicked : function (ev) {
-                var checkbox = $("#" + ev.target.parentElement.id).find(':checkbox')
+
+         selectClicked : function (ev) {
+            var checkbox = $("#" + ev.target.parentElement.id).find(':checkbox')
                 if (checkbox[0] != undefined) {
                   if (checkbox[0].checked){
                     checkbox[0].checked = '';
@@ -324,11 +434,161 @@
                     checkbox[0].checked = 'selected';
                 }
             }
-        }
+            this.showTools();
+        },
+
+        // show toolbar in case something is selected
+        showTools : function(){
+            var checkedValues = $('#folder_table').find(':checked');
+            if(checkedValues.length > 0){
+                $('#toolbtn_bulk_import').show();
+                $('#toolbtn_dl').show();
+            } else {
+                $('#toolbtn_bulk_import').hide();
+                $('#toolbtn_dl').hide();
+            }
+
+        },
+
+        // show bulk import modal
+        modalBulkImport : function(){
+            var self = this;
+            // fetch histories
+            var histories = new GalaxyHistories();
+            histories.fetch({
+                    success: function (histories){
+                        // make modal
+                        var history_modal_tmpl =  _.template(self.templateBulkImportInModal(), {histories : histories.models});
+                        self.modal = new mod_modal.GalaxyModal({
+                            title   : 'Import into History',
+                            body    : history_modal_tmpl,
+                            buttons : {
+                                'Import' : function() {self.importAllIntoHistory()},
+                                'Close'  : function() {self.modal.hide()}
+                            }
+                        });
+                        // show the prepared modal
+                        self.modal.show();
+                    }
+                });
+        },
+
+        // import all selected datasets into history
+        importAllIntoHistory : function (){
+            //disable the button
+            this.modal.disableButton('Import');
+
+            var history_id = $("select[name=dataset_import_bulk] option:selected").val();
+            var history_name = $("select[name=dataset_import_bulk] option:selected").text();
+
+            var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+            var progress_bar_tmpl = _.template(this.templateProgressBar(), { history_name : history_name });
+            $(this.modal.elMain).find('.modal-body').html(progress_bar_tmpl);
+            
+            // init the progress bar
+            var progressStep = 100 / dataset_ids.length;
+            this.initProgress(progressStep);
+
+            // prepare the dataset objects to be imported
+            var datasets_to_import = [];
+            for (var i = dataset_ids.length - 1; i >= 0; i--) {
+                library_dataset_id = dataset_ids[i];
+                var historyItem = new HistoryItem();
+                var self = this;
+                historyItem.url = historyItem.urlRoot + history_id + '/contents';
+                historyItem.content = library_dataset_id;
+                historyItem.source = 'library';
+                datasets_to_import.push(historyItem);
+            };
+
+            // call the recursive function to call ajax one after each other
+            this.chainCall(datasets_to_import);
+        },
+
+        chainCall: function(history_item_set){
+            var self = this;
+            var popped_item = history_item_set.pop();
+            if (typeof popped_item === "undefined") {
+                self.modal.showNotification('All datasets imported', 3000, '#e1f4e0', '#32a427');
+                // enable button again
+                self.modal.enableButton('Import');
+                return
+            }
+                var promise = $.when(popped_item.save({content: popped_item.content, source: popped_item.source})).done(function(a1){
+                        self.updateProgress();
+                        responses.push(a1);
+                        self.chainCall(history_item_set);
+                    });
+        },
+
+        initProgress: function(progressStep){
+            this.progress = 0;
+            this.progressStep = progressStep;
+        },
+        updateProgress: function(){
+            this.progress += this.progressStep;
+            $('.progress-bar').width(Math.round(this.progress) + '%');
+            txt_representation = Math.round(this.progress) + '% Complete';
+            $('.completion_span').text(txt_representation);
+        },
+
+      // progress bar 
+      templateProgressBar : function (){
+        var tmpl_array = [];
+
+        tmpl_array.push('<div class="import_text">');
+        tmpl_array.push('Importing selected datasets to history <b><%= _.escape(history_name) %></b>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('<div class="progress">');
+        tmpl_array.push('   <div class="progress-bar" role="progressbar" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100" style="width: 00%;">');
+        tmpl_array.push('       <span class="completion_span">0% Complete</span>');
+        tmpl_array.push('   </div>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('');
+
+        return tmpl_array.join('');
+      },
+
+      // download selected datasets
+      download : function(folder_id, format){
+        var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+
+        var url = '/api/libraries/datasets/download/' + format;
+        var data = {'ldda_ids' : dataset_ids};
+        this.processDownload(url, data);
+      },
+
+      // create hidden form and submit through POST to initialize download
+      processDownload: function(url, data, method){
+        //url and data options required
+        if( url && data ){ 
+                //data can be string of parameters or array/object
+                data = typeof data == 'string' ? data : $.param(data);
+                //split params into form inputs
+                var inputs = '';
+                $.each(data.split('&'), function(){ 
+                        var pair = this.split('=');
+                        inputs+='<input type="hidden" name="'+ pair[0] +'" value="'+ pair[1] +'" />'; 
+                });
+                //send request
+                $('<form action="'+ url +'" method="'+ (method||'post') +'">'+inputs+'</form>')
+                .appendTo('body').submit().remove();
+        };
+      }
+
     });
 
-
-// galaxy library
+// galaxy library view
 var GalaxyLibraryview = Backbone.View.extend({
     el: '#center',
 
@@ -343,9 +603,10 @@
     // template
     template_library_list : function (){
         tmpl_array = [];
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; overflow: auto !important; ">');
 
         tmpl_array.push('');
-        tmpl_array.push('<h1>Welcome to the data libraries</h1>');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to Martin.</h3>');
         tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary icon-file ">New Library</a>');
         tmpl_array.push('<table class="table table-striped">');
         tmpl_array.push('   <thead>');
@@ -353,30 +614,27 @@
         tmpl_array.push('     <th>description</th>');
         tmpl_array.push('     <th>synopsis</th> ');
         tmpl_array.push('     <th>model type</th> ');
-        // tmpl_array.push('     <th>id</th> ');
         tmpl_array.push('   </thead>');
         tmpl_array.push('   <tbody>');
         tmpl_array.push('       <% _.each(libraries, function(library) { %>');
-            tmpl_array.push('           <tr>');
-            tmpl_array.push('               <td><a href="#/folders/<%- library.id %>"><%- library.get("name") %></a></td>');
-            tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
-            tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
-            tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
-        // tmpl_array.push('               <td><a href="#/folders/<%- library.id %>"><%= _.escape(library.get("id")) %></a></td>');
+        tmpl_array.push('           <tr>');
+        tmpl_array.push('               <td><a href="#/folders/<%- library.get("root_folder_id") %>"><%- library.get("name") %></a></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
         tmpl_array.push('           </tr>');
         tmpl_array.push('       <% }); %>');
-        tmpl_array.push('');
-        tmpl_array.push('');
-        tmpl_array.push('');
-        tmpl_array.push('');
         tmpl_array.push('   </tbody>');
         tmpl_array.push('</table>');
-
+        
+        tmpl_array.push('</div>');
         return tmpl_array.join('');
     },
 
     // render
     render: function () {
+        //hack to show scrollbars
+        $("#center").css('overflow','auto');
         var that = this;
         // if (typeof libraries === "undefined") {
           libraries = new Libraries();
@@ -394,12 +652,11 @@
 
     // show/hide create library modal
     show_library_modal : function (e){
-        // prevent default
+        // prevent default, may be unnecessary
         e.preventDefault();
         
         // create modal
         if (!this.modal){
-            // make modal
             var self = this;
             this.modal = new mod_modal.GalaxyModal(
             {
@@ -412,9 +669,11 @@
             });
         }
         
-        // show modal
+        // show prepared modal
         this.modal.show();
     },
+
+    // create the new library from modal
     create_new_library_event: function(){
         var libraryDetails = this.serialize_new_library();
         var library = new Library();
@@ -428,11 +687,15 @@
     });
         return false;
     },
+
+    // clear the library modal once saved
     clear_library_modal : function(){
         $("input[name='Name']").val('');
         $("input[name='Description']").val('');
         $("input[name='Synopsis']").val('');
     },
+
+    // serialize data from the form
     serialize_new_library : function(){
         return {
             name: $("input[name='Name']").val(),
@@ -440,10 +703,12 @@
             synopsis: $("input[name='Synopsis']").val()
         };
     },
-    // load html template
+
+    // template for new library modal
     template_new_library: function()
     {
         tmpl_array = [];
+
         tmpl_array.push('<div id="new_library_modal">');
         tmpl_array.push('<form>');
         tmpl_array.push('<input type="text" name="Name" value="" placeholder="Name">');
@@ -452,10 +717,8 @@
         tmpl_array.push('');
         tmpl_array.push('</form>');
         tmpl_array.push('</div>');
+
         return tmpl_array.join('');
-
-
-        // return  '<div id="'+ id +'"></div>';
     }
 });
 
@@ -473,19 +736,28 @@
         library_router.on('route:libraries', function() {
           // render libraries list
           galaxyLibraryview.render();
-      })           
+      });
 
         library_router.on('route:folder_content', function(id) {
           // render folder's contents
           folderContentView.render({id: id});
+      });    
 
-      })      
+       library_router.on('route:download', function(folder_id, format) {
+          // send download stream
+          if (typeof folderContentView === 'undefined'){
+            alert('you cant touch this!');
+          // } else if (folderContentView.modal !== null){
+            // folderContentView.download(folder_id, format);
+          } else if ($('#center').find(':checked').length === 0) { // coming from outside of the library app
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+            // TODO show message of redirection
+          } else {
+            folderContentView.download(folder_id, format);
+            library_router.navigate('folders/' + folder_id, {trigger: false, replace: true});
+          }
 
-        // library_router.on('route:show_library_modal', function() {
-        //   // render folder's contents
-        //   galaxyLibraryview.show_library_modal();
-        
-        // })
+      });      
 
 Backbone.history.start();   
 
diff -r 4e53d7b759d1999758a30611e5d897e3e1d20782 -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -20,9 +20,28 @@
     
     // initialize
     initialize : function(options) {
+        self = this;
         // create
         if (options)
             this.create(options);
+            // bind the ESC key to hide() function
+            $(document).on('keyup', function(event){
+                if (event.keyCode == 27) { self.hide(); }
+            })
+            // bind the click anywhere to hide() function...
+            $('html').on('click', function(event){
+                self.hide();
+            })
+            // ...but don't hide if the click is on modal content
+            $('.modal-content').on('click', function(event){
+                event.stopPropagation();
+            })
+    },
+
+    // destroy
+    destroy : function(){
+        this.hide();
+        $('.modal').html('');
     },
 
     // adds and displays a new frame/window
https://bitbucket.org/galaxy/galaxy-central/commits/f466834f693f/
Changeset:   f466834f693f
Branch:      data_library
User:        martenson
Date:        2013-12-10 19:01:19
Summary:     improved styles for libraries, added features to modal window, API tweaks
Affected #:  5 files
diff -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 -r f466834f693ff7ae0f60cf121d97f67a5338ea95 lib/galaxy/model/mapping.py
--- a/lib/galaxy/model/mapping.py
+++ b/lib/galaxy/model/mapping.py
@@ -1988,15 +1988,6 @@
     table = self.table
     trans = conn.begin()
     try:
-        
-#         log.debug("XXXXXXXXXXXXXXX NEW GENERATOOOOOR")
-#         next_hid = table.update() \
-#         .returning( table.c.hid_counter ) \
-#         .where( table.c.id == self.id ) \
-#         .values( hid_counter = hid_counter + 1 )
-#         log.debug('XXXXXXXXXXXXXXXXXXXXXXXXXXX NEXT HID: ' + str(next_hid))
-        
-    
         current_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
         next_hid = current_hid + 1
         table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid ) )
diff -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 -r f466834f693ff7ae0f60cf121d97f67a5338ea95 lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -46,17 +46,6 @@
         """
         folder_container = []
         current_user_roles = trans.get_current_user_roles()
-        
-#         log.debug( 'XXXXXXXXXXXXXXXXXXXXX GOT id: ' + folder_id )
-#         log.debug( 'XXXXXXXXXXXXXXXXXXXXX decode id: ' + folder_id[1:] )
-#         log.debug( 'XXXXXXXXXXXXXXXXXXXXX call decode: ' + str(trans.security.decode_id( folder_id[1:] )) )
-
-#         try:
-#             decoded_folder_id = trans.security.decode_id( folder_id[-16:] )
-#         except TypeError:
-#             trans.response.status = 400
-#             return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
-#         log.debug( 'XXXXXXXXXXXXXXXXXXXXX decoded id: ' + decoded_folder_id )
 
         if ( folder_id.startswith( 'F' ) ):
             try:
@@ -67,9 +56,6 @@
 
         try:
             folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( decoded_folder_id )
-#             log.debug( 'XXXXXXXXXXXXXXXXXXXXX FOLDER id: ' + str(folder.id) )
-#             log.debug( 'XXXXXXXXXXXXXXXXXXXXX FOLDER name: ' + str(folder.name) )
-#             parent_library = folder.parent_library
         except:
             folder = None
             log.error( "FolderContentsController.index: Unable to retrieve folder with ID: %s" % folder_id )
@@ -92,17 +78,9 @@
             path_to_root = []
             # We are almost in root
             if folder.parent_id is None:
-                
-#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ALMOST ROOT FOLDER! ADDING ID: " + str( folder.id ) )
-#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ALMOST ROOT FOLDER! ADDING NAME: " + str( folder.name ) )
                 path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
-#                 upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_library.id )
-#                 path_to_root.append( ( upper_folder.id, upper_folder.name ) )
             else:
             # We add the current folder and traverse up one folder.
-#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX folder.parent_id " + str( folder.parent_id ) )
-#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ALMOST ROOT FOLDER! ADDING ID: " + str( folder.id ) )
-#                 log.debug( "XXXXXXXXXXXXXXXXXXXXXXX ADDING THIS FOLDER AND TRAVERSING UP:  " + str( folder.name ) )
                 path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
                 upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_id )
                 path_to_root.extend( build_path( upper_folder ) )
@@ -113,25 +91,40 @@
         folder_container.append( dict( full_path = full_path ) )
         
         folder_contents = []
+        time_updated = ''
+        time_created = ''
         # Go through every item in the folder and include its meta-data.
         for content_item in self.load_folder_contents( trans, folder ):
+#             rval = content_item.to_dict()
             return_item = {}
             encoded_id = trans.security.encode_id( content_item.id )
+            time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+            time_created = content_item.create_time.strftime( "%Y-%m-%d %I:%M %p" )
+#             log.debug('XXXXXXXXXXXXXXXXXXX api type: ' + str(content_item.api_type))
+#             log.debug('XXXXXXXX ALL: ' + str(content_item.__dict__))
             
             # For folder return also hierarchy values
             if content_item.api_type == 'folder':
-#                 encoded_parent_library_id = trans.security.encode_id( content_item.parent_library.id )
                 encoded_id = 'F' + encoded_id
-#                 if content_item.parent_id is not None: # Return folder's parent id for browsing back.
-#                     encoded_parent_id = 'F' + trans.security.encode_id( content_item.parent_id )
-                last_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
-                return_item.update ( dict ( item_count = content_item.item_count, last_updated = last_updated ) )
-            
+#                 time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+                return_item.update ( dict ( item_count = content_item.item_count ) )
+
+            if content_item.api_type == 'file':
+#                 log.debug('XXXXX content item class: ' + str(content_item.__class__))
+                library_dataset_dict = content_item.to_dict()
+                library_dataset_dict['data_type']
+                library_dataset_dict['file_size']
+                library_dataset_dict['date_uploaded']
+                return_item.update ( dict ( data_type = library_dataset_dict['data_type'],
+                                            file_size = library_dataset_dict['file_size'],
+                                            date_uploaded = library_dataset_dict['date_uploaded'] ) )
+
             # For every item return also the default meta-data
             return_item.update( dict( id = encoded_id,
                                type = content_item.api_type,
-                               name = content_item.name
-                               
+                               name = content_item.name,
+                               time_updated = time_updated,
+                               time_created = time_created
                                 ) )
             folder_contents.append( return_item )
         # Put the data in the container
diff -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 -r f466834f693ff7ae0f60cf121d97f67a5338ea95 static/scripts/galaxy.library.js
--- a/static/scripts/galaxy.library.js
+++ b/static/scripts/galaxy.library.js
@@ -6,8 +6,16 @@
 var view            = null;
 var library_router  = null;
 var responses = [];
-var KEYCODE_ENTER = 13;
-var KEYCODE_ESC = 27;
+
+
+// load required libraries
+require([
+    // load js libraries
+    'utils/galaxy.css',
+    ], function(css){
+        // load css
+        css.load_file("static/style/library.css");
+});
 
 // dependencies
 define(["galaxy.modal", "galaxy.master"], function(mod_modal, mod_master) {
@@ -115,20 +123,18 @@
 
         // CONTAINER
         tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; ">');
-        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to Martin.</h3>');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
 
         // TOOLBAR
         tmpl_array.push('<div id="library_folder_toolbar" >');
-        tmpl_array.push('   <button id="toolbtn_create_folder" class="btn btn-primary" type="button">new folder</button>');
-        tmpl_array.push('   <button id="toolbtn_bulk_import" style="display: none; margin-left: 0.5em;" type="button">Import into History</button>');
+        tmpl_array.push('   <button title="Create New Folder" id="toolbtn_create_folder" class="btn btn-primary" type="button"><span class="fa fa-icon-plus"></span><span class="fa fa-icon-folder-close"></span> folder</button>');
+        tmpl_array.push('   <button id="toolbtn_bulk_import" class="btn btn-primary" style="display: none; margin-left: 0.5em;" type="button"><span class="fa fa-icon-external-link"></span> to history</button>');
         
         tmpl_array.push('   <div id="toolbtn_dl" class="btn-group" style="margin-left: 0.5em; display: none; ">');
-        tmpl_array.push('       <button id="drop_toggle" type="button" class="btn btn-default dropdown-toggle" data-toggle="dropdown">');
-        tmpl_array.push('       Download <span class="caret"></span>');
+        tmpl_array.push('       <button id="drop_toggle" type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown">');
+        tmpl_array.push('       <span class="fa fa-icon-download"></span> download <span class="caret"></span>');
         tmpl_array.push('       </button>');
         tmpl_array.push('       <ul class="dropdown-menu" role="menu">');
-        // tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/uncompressed">uncompressed</a></li>');
-        // tmpl_array.push('          <li class="divider"></li>');
         tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tgz">.tar.gz</a></li>');
         tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tbz">.tar.bz</a></li>');
         tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/zip">.zip</a></li>');
@@ -138,41 +144,64 @@
         tmpl_array.push('</div>');
 
         // BREADCRUMBS
-        tmpl_array.push('<a href="#">Libraries</a><b>|</b> ');
-        tmpl_array.push('<% _.each(path, function(path_item) { %>'); //breadcrumb
+        tmpl_array.push('<div class="library_breadcrumb">');
+        tmpl_array.push('<a title="Return to the list of libraries" href="#">Libraries</a><b>|</b> ');
+        tmpl_array.push('<% _.each(path, function(path_item) { %>');
         tmpl_array.push('<% if (path_item[0] != id) { %>');
-        tmpl_array.push('<a href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a><b>|</b> ');
+        tmpl_array.push('<a title="Return to this folder" href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a><b>|</b> ');
         tmpl_array.push('<% } else { %>');
-        tmpl_array.push('<%- path_item[1] %>');
+        tmpl_array.push('<span title="You are in this folder"><%- path_item[1] %></span>');
         tmpl_array.push('<% } %>');
         tmpl_array.push('<% }); %>');
+        tmpl_array.push('</div>');
 
-        // FODLER CONTENT
-        tmpl_array.push('<table id="folder_table" class="table table-hover table-condensed">');
+        // FOLDER CONTENT
+        tmpl_array.push('<table id="folder_table" class="table table-condensed">');
         tmpl_array.push('   <thead>');
         tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
+        tmpl_array.push('       <th class="button_heading">view</th>');
         tmpl_array.push('       <th>name</th>');
-        tmpl_array.push('       <th>type</th>');
+        tmpl_array.push('       <th>data type</th>');
+        tmpl_array.push('       <th>size</th>');
+        tmpl_array.push('       <th>date</th>');
         tmpl_array.push('   </thead>');
         tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td><button title="Go to parent folder" type="button" data-id="<%- upper_folder_id %>" class="btn_open_folder btn btn-default btn-xs">');        
+        tmpl_array.push('       <span class="fa fa-icon-arrow-up"></span> .. go up</td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       </tr>');        
         tmpl_array.push('       <% _.each(items, function(content_item) { %>');
-        tmpl_array.push('       <tr class="folder_row" id="<%- content_item.id %>">');
-        tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>'); //folder
-        tmpl_array.push('               <td style="text-align: center; "></td>');
-        tmpl_array.push('               <td><a href="#/folders/<%- content_item.id %>"><%- content_item.get("name") %></a>');
-        tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>'); //empty folder
+        tmpl_array.push('       <tr class="folder_row light" id="<%- content_item.id %>">');
+        tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>'); // folder
+        tmpl_array.push('               <td></td>');
+        tmpl_array.push('               <td><button title="Open this folder" type="button" data-id="<%- content_item.id %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- content_item.get("name") %>');
+        tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>'); // empty folder
         tmpl_array.push('           <span class="muted">(empty folder)</span>');
         tmpl_array.push('           <% } %>');
         tmpl_array.push('           </td>');
+        tmpl_array.push('           <td>folder</td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("item_count")) %> item(s)</td>'); // size
         tmpl_array.push('           <% } else {  %>');
         tmpl_array.push('           <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
-        tmpl_array.push('           <td><a class="library-dataset" href="#"><%- content_item.get("name") %></a></td>'); //dataset
+        tmpl_array.push('       <td>');
+        tmpl_array.push('       <button title="See details of this dataset" type="button" class="library-dataset btn btn-default btn-xs">');
+        tmpl_array.push('       <span class="fa fa-icon-eye-open"></span> details');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       </td>');
+        tmpl_array.push('           <td><%- content_item.get("name") %></td>'); // dataset
+        tmpl_array.push('           <td><%= _.escape(content_item.get("data_type")) %></td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("readable_size")) %></td>'); // size
         tmpl_array.push('           <% } %>  ');
-        tmpl_array.push('           <td><%= _.escape(content_item.get("type")) %></td>');
+        tmpl_array.push('           <td><%= _.escape(content_item.get("time_updated")) %></td>'); // time updated
         tmpl_array.push('       </tr>');
         tmpl_array.push('       <% }); %>');
         tmpl_array.push('       ');
-        tmpl_array.push('       ');
         tmpl_array.push('   </tbody>');
         tmpl_array.push('</table>');
 
@@ -292,7 +321,9 @@
             'click .folder_row' : 'selectClicked',
             'click #toolbtn_bulk_import' : 'modalBulkImport',
             'click #toolbtn_dl' : 'bulkDownload',
-            'click .library-dataset' : 'showDatasetDetails'
+            'click .library-dataset' : 'showDatasetDetails',
+            'click #toolbtn_create_folder' : 'createFolderModal',
+            'click .btn_open_folder' : 'navigateToFolder'
         },
 
       //render the folder view
@@ -308,22 +339,52 @@
 
         folderContainer.fetch({
           success: function (container) {
-              // folderContainer.attributes.folder = container.attributes.folder;
-              var template = _.template(that.templateFolder(), {path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id});
+
+            // prepare nice size strings
+              for (var i = 0; i < folderContainer.attributes.folder.models.length; i++) {
+                  var model = folderContainer.attributes.folder.models[i]
+                  if (model.get('type') === 'file'){
+                    model.set('readable_size', that.size_to_string(model.get('file_size')))
+                  }
+              };
+
+              // find the upper id
+              var path = folderContainer.full_path;
+              var upper_folder_id;
+              if (path.length === 1){ // library is above us
+                upper_folder_id = 0;
+              } else {
+                upper_folder_id = path[path.length-2][0];
+              }
+
+              var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id, upper_folder_id: upper_folder_id });
+              // var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id });
               that.$el.html(template);
             }
           })
         },
 
+      // handles the click on 'open' and 'upper' folder icons
+      navigateToFolder : function(event){
+        var folder_id = $(event.target).attr('data-id');
+        if (typeof folder_id === 'undefined') {
+            return false;
+        } else if (folder_id === '0'){
+            library_router.navigate('#', {trigger: true, replace: true});
+        } else {
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+        }
+      },
+
       //show modal with current dataset info
-      showDatasetDetails : function(e){
+      showDatasetDetails : function(event){
         // prevent default
-        e.preventDefault();
+        event.preventDefault();
         
 //TODO check whether we already have the data
 
         //load the ID of the row
-        var id = $(e.target).parent().parent().attr('id');
+        var id = $(event.target).parent().parent().attr('id');
         
         //create new item
         var item = new Item();
@@ -356,7 +417,7 @@
                 buttons : {
                     'Import' : function() { self.importCurrentIntoHistory() },
                     'Download' : function() { self.downloadCurrent() },
-                    'Close'  : function() { self.modal.hide(); $('.modal').html(''); self.modal = null; } // TODO refill nicely modal with data
+                    'Close'  : function() { self.modal.hide(); $('.modal').remove(); self.modal = null; } // TODO refill nicely modal with data
                 }
             });
             $(".peek").html(item.get("peek"));
@@ -405,12 +466,12 @@
 
             // save the dataset into selected history
             historyItem.save({ content : library_dataset_id, source : 'library' }, { success : function(){
-                self.modal.showNotification('Dataset imported', 3000, '#e1f4e0', '#32a427');
+                self.modal.showNotification('Dataset imported', 3000, 'success');
                 //enable the buttons
                 self.modal.enableButton('Import');
                 self.modal.enableButton('Download');
             }, error : function(){
-                self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, '#f4e0e1', '#a42732');
+                self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, 'error');
                 //enable the buttons
                 self.modal.enableButton('Import');
                 self.modal.enableButton('Download');                
@@ -418,20 +479,32 @@
         });
         },
 
-        selectAll : function (ev) {
-           var selected = ev.target.checked;
+        // select all datasets 
+        selectAll : function (event) {
+           var selected = event.target.checked;
                  // Iterate each checkbox
                  $(':checkbox').each(function () { this.checked = selected; });
                  this.showTools();
              },
 
-         selectClicked : function (ev) {
-            var checkbox = $("#" + ev.target.parentElement.id).find(':checkbox')
+         // click checkbox on folder click
+         selectClicked : function (event) {
+            var checkbox = $("#" + event.target.parentElement.id).find(':checkbox')
                 if (checkbox[0] != undefined) {
                   if (checkbox[0].checked){
                     checkbox[0].checked = '';
+                    // $(event.target.parentElement).css('background-color', '').css('color', '');
+                    $(event.target.parentElement).removeClass('dark');
+                    $(event.target.parentElement).find('a').removeClass('dark');
+                    $(event.target.parentElement).addClass('light');
+                    $(event.target.parentElement).find('a').addClass('light');
                 } else {
                     checkbox[0].checked = 'selected';
+                    $(event.target.parentElement).removeClass('light');
+                    $(event.target.parentElement).find('a').removeClass('light');
+                    $(event.target.parentElement).addClass('dark');
+                    $(event.target.parentElement).find('a').addClass('dark');
+                    // $(event.target.parentElement).css('background-color', '#8389a1').css('color', 'white');
                 }
             }
             this.showTools();
@@ -464,7 +537,7 @@
                             body    : history_modal_tmpl,
                             buttons : {
                                 'Import' : function() {self.importAllIntoHistory()},
-                                'Close'  : function() {self.modal.hide()}
+                                'Close'  : function() {self.modal.hide(); $('.modal').remove(); self.modal = null;}
                             }
                         });
                         // show the prepared modal
@@ -514,7 +587,7 @@
             var self = this;
             var popped_item = history_item_set.pop();
             if (typeof popped_item === "undefined") {
-                self.modal.showNotification('All datasets imported', 3000, '#e1f4e0', '#32a427');
+                self.modal.showNotification('All datasets imported', 3000, 'success');
                 // enable button again
                 self.modal.enableButton('Import');
                 return
@@ -584,6 +657,11 @@
                 $('<form action="'+ url +'" method="'+ (method||'post') +'">'+inputs+'</form>')
                 .appendTo('body').submit().remove();
         };
+      },
+
+      // shows modal for creating folder
+      createFolderModal: function(){
+        alert('creating folder');
       }
 
     });
@@ -606,11 +684,12 @@
         tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; overflow: auto !important; ">');
 
         tmpl_array.push('');
-        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to Martin.</h3>');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
         tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary icon-file ">New Library</a>');
-        tmpl_array.push('<table class="table table-striped">');
+        tmpl_array.push('<table class="table table-condensed">');
         tmpl_array.push('   <thead>');
-        tmpl_array.push('    <th>name</th>');
+        tmpl_array.push('     <th class="button_heading"></th>');
+        tmpl_array.push('     <th>name</th>');
         tmpl_array.push('     <th>description</th>');
         tmpl_array.push('     <th>synopsis</th> ');
         tmpl_array.push('     <th>model type</th> ');
@@ -618,7 +697,9 @@
         tmpl_array.push('   <tbody>');
         tmpl_array.push('       <% _.each(libraries, function(library) { %>');
         tmpl_array.push('           <tr>');
-        tmpl_array.push('               <td><a href="#/folders/<%- library.get("root_folder_id") %>"><%- library.get("name") %></a></td>');
+        tmpl_array.push('               <td><button title="Open this library" type="button" data-id="<%- library.get("root_folder_id") %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- library.get("name") %></td>');
         tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
         tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
         tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
@@ -651,12 +732,11 @@
     modal : null,
 
     // show/hide create library modal
-    show_library_modal : function (e){
-        // prevent default, may be unnecessary
-        e.preventDefault();
+    show_library_modal : function (event){
+        event.preventDefault();
+        event.stopPropagation();
         
         // create modal
-        if (!this.modal){
             var self = this;
             this.modal = new mod_modal.GalaxyModal(
             {
@@ -667,7 +747,6 @@
                     'Close'  : function() {self.modal.hide()}
                 }
             });
-        }
         
         // show prepared modal
         this.modal.show();
@@ -676,6 +755,7 @@
     // create the new library from modal
     create_new_library_event: function(){
         var libraryDetails = this.serialize_new_library();
+        var valid = this.validate_new_library(libraryDetails);
         var library = new Library();
         var self = this;
         library.save(libraryDetails, {
@@ -683,7 +763,10 @@
             self.modal.hide();
             self.clear_library_modal();
             self.render();
-        }
+          },
+          error: function(){
+            self.modal.showNotification('An error occured', 5000, 'error');
+            }
     });
         return false;
     },
@@ -704,6 +787,11 @@
         };
     },
 
+    validate_new_library: function(library){
+
+    },
+
+
     // template for new library modal
     template_new_library: function()
     {
@@ -714,7 +802,6 @@
         tmpl_array.push('<input type="text" name="Name" value="" placeholder="Name">');
         tmpl_array.push('<input type="text" name="Description" value="" placeholder="Description">');
         tmpl_array.push('<input type="text" name="Synopsis" value="" placeholder="Synopsis">');
-        tmpl_array.push('');
         tmpl_array.push('</form>');
         tmpl_array.push('</div>');
 
diff -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 -r f466834f693ff7ae0f60cf121d97f67a5338ea95 static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -1,6 +1,5 @@
-/*
-    galaxy modal
-*/
+// galaxy modal
+
 
 // dependencies
 define(["libs/backbone/backbone-relational"], function() {
@@ -24,24 +23,48 @@
         // create
         if (options)
             this.create(options);
-            // bind the ESC key to hide() function
-            $(document).on('keyup', function(event){
-                if (event.keyCode == 27) { self.hide(); }
-            })
-            // bind the click anywhere to hide() function...
-            $('html').on('click', function(event){
-                self.hide();
-            })
-            // ...but don't hide if the click is on modal content
-            $('.modal-content').on('click', function(event){
-                event.stopPropagation();
-            })
+
+        this.bindClick(event, self);
     },
 
+    // bind the click-to-hide function
+    bindClick: function(event, that) {
+        // bind the ESC key to hide() function
+        $(document).on('keyup', function(event){
+            if (event.keyCode == 27) { self.hide(); }
+        })
+        // bind the click anywhere to hide() function...
+        $('html').on('click', function(event){
+            that.hide();
+        })
+        // ...but don't hide if the click is on modal content
+        $('.modal-content').on('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+    // unbind the click-to-hide function
+    unbindClick: function(event, that){
+        // bind the ESC key to hide() function
+        $(document).off('keyup', function(event){
+            if (event.keyCode == 27) { that.hide(); }
+        })
+        // unbind the click anywhere to hide() function...
+        $('html').off('click', function(event){
+            that.hide();
+        })
+        // ...but don't hide if the click is on modal content
+        $('.modal-content').off('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+
     // destroy
     destroy : function(){
         this.hide();
-        $('.modal').html('');
+        this.unbindClick();
+        $('.modal').remove();
     },
 
     // adds and displays a new frame/window
@@ -152,16 +175,28 @@
     },
 
     // show notification
-    showNotification : function(message, duration, bgColor, txtColor) {
+    showNotification : function(message, duration, type) {
         // defaults
         var duration = typeof duration !== 'undefined' ? duration : 1500;
-        var bgColor = typeof bgColor !== 'undefined' ? bgColor : "#F4E0E1";
-        var txtColor = typeof txtColor !== 'undefined' ? txtColor : "#A42732";
+        // var bgColor = typeof bgColor !== 'undefined' ? bgColor : "#F4E0E1";
+        // var txtColor = typeof txtColor !== 'undefined' ? txtColor : "#A42732";
+        var bgColor;
+        var txtColor;
+
+        if (type === 'error'){
+            bgColor = '#f4e0e1';
+            txtColor = '#a42732';
+        // } else if (type === 'success'){
+        } else { // success is default
+            bgColor = '#e1f4e0';
+            txtColor = '#32a427'; 
+        }
 
         var HTMLmessage = "<div class='notification-message' style='text-align:center; line-height:16px; '> " + message + " </div>";
         this.$notification.html("<div id='notification-bar' style='display:none; float: right; height: 16px; width:100%; background-color: " + bgColor + "; z-index: 100; color: " + txtColor + ";border-bottom: 1px solid " + txtColor + ";'>" + HTMLmessage + "</div>");
 
         var self = this;
+        
         /*animate the bar*/
         $('#notification-bar').slideDown(function() {
             setTimeout(function() {
diff -r 44399512777bdefb39ea3c99cc04b3f7947c5d33 -r f466834f693ff7ae0f60cf121d97f67a5338ea95 static/style/src/less/library.less
--- a/static/style/src/less/library.less
+++ b/static/style/src/less/library.less
@@ -29,6 +29,50 @@
     background-color: @table-bg-accent;
 }
 
+
+tr.light td
+{
+    background-color: white;
+    color: black;
+}
+tr.light:hover td
+{
+    background-color: #f5f5f5;
+    color: #8389a1;
+}
+tr.dark td
+{
+    background-color: #8389a1;
+    color: white;
+}
+tr.dark:hover td
+{
+    background-color: #bbbfd0;
+    color: white;
+}
+a.dark:hover
+{
+    color: yellow;
+    // text-decoration: none;
+}
+a.dark
+{
+    color: white;
+    // text-decoration: none;
+}
+th.button_heading
+{
+    width: 7em;
+}
+div.library_breadcrumb{
+    padding-top: 0.8em;
+    padding-bottom: 0.8em;
+}
+div.library_breadcrumb a:hover{
+    color:green;
+}
+
+
 img.expanderIcon {
     padding-right: 4px;
 }
https://bitbucket.org/galaxy/galaxy-central/commits/e2b8d3481d66/
Changeset:   e2b8d3481d66
User:        martenson
Date:        2013-12-11 00:02:36
Summary:     Merge data libraries branch to default
Affected #:  19 files
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 .hgignore
--- a/.hgignore
+++ b/.hgignore
@@ -116,3 +116,6 @@
 *.rej
 *~
 
+
+syntax: regexp
+^static/AAA_scratch$
\ No newline at end of file
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 lib/galaxy/model/__init__.py
--- a/lib/galaxy/model/__init__.py
+++ b/lib/galaxy/model/__init__.py
@@ -1827,7 +1827,7 @@
 class Library( object, Dictifiable ):
     permitted_actions = get_permitted_actions( filter='LIBRARY' )
     dict_collection_visible_keys = ( 'id', 'name' )
-    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis' )
+    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis', 'root_folder_id' )
     def __init__( self, name=None, description=None, synopsis=None, root_folder=None ):
         self.name = name or "Unnamed library"
         self.description = description
@@ -1894,7 +1894,7 @@
         return name
 
 class LibraryFolder( object, Dictifiable ):
-    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build' )
+    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build', 'update_time' )
     def __init__( self, name=None, description=None, item_count=0, order_id=None ):
         self.name = name or "Unnamed folder"
         self.description = description
@@ -2060,6 +2060,7 @@
                      genome_build = ldda.dbkey,
                      misc_info = ldda.info,
                      misc_blurb = ldda.blurb,
+                     peek = ( lambda ldda: ldda.display_peek() if ldda.peek and ldda.peek != 'no peek' else None )( ldda ),
                      template_data = template_data )
         if ldda.dataset.uuid is None:
             rval['uuid'] = None
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 lib/galaxy/model/mapping.py
--- a/lib/galaxy/model/mapping.py
+++ b/lib/galaxy/model/mapping.py
@@ -1878,8 +1878,9 @@
     table = self.table
     trans = conn.begin()
     try:
-        next_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
-        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid + 1 ) )
+        current_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
+        next_hid = current_hid + 1
+        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid ) )
         trans.commit()
         return next_hid
     except:
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 lib/galaxy/webapps/galaxy/api/datasets.py
--- a/lib/galaxy/webapps/galaxy/api/datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/datasets.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a dataset.
+API operations on the contents of a history dataset.
 """
 from galaxy import web
 from galaxy.visualization.data_providers.genome import FeatureLocationIndexDataProvider
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a library.
+API operations on the contents of a folder.
 """
 import logging, os, string, shutil, urllib, re, socket
 from cgi import escape, FieldStorage
@@ -11,67 +11,125 @@
 log = logging.getLogger( __name__ )
 
 class FolderContentsController( BaseAPIController, UsesLibraryMixin, UsesLibraryMixinItems ):
+    """
+    Class controls retrieval, creation and updating of folder contents.
+    """
+
+    def load_folder_contents( self, trans, folder ):
+        """
+        Loads all contents of the folder (folders and data sets) but only in the first level.
+        """
+        current_user_roles = trans.get_current_user_roles()
+        is_admin = trans.user_is_admin()
+        content_items = []
+        for subfolder in folder.active_folders:
+            if not is_admin:
+                can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
+            if (is_admin or can_access) and not subfolder.deleted:
+                subfolder.api_type = 'folder'
+                content_items.append( subfolder )
+        for dataset in folder.datasets:
+            if not is_admin:
+                can_access = trans.app.security_agent.can_access_dataset( current_user_roles, dataset.library_dataset_dataset_association.dataset )
+            if (is_admin or can_access) and not dataset.deleted:
+                dataset.api_type = 'file'
+                content_items.append( dataset )
+        return content_items
 
     @web.expose_api
     def index( self, trans, folder_id, **kwd ):
         """
         GET /api/folders/{encoded_folder_id}/contents
         Displays a collection (list) of a folder's contents (files and folders).
-        The /api/library_contents/{encoded_library_id}/contents
-        lists everything in a library recursively, which is not what
-        we want here. We could add a parameter to use the recursive
-        style, but this is meant to act similar to an "ls" directory listing.
+        Encoded folder ID is prepended with 'F' if it is a folder as opposed to a data set which does not have it.
+        Full path is provided as a separate object in response providing data for breadcrumb path building.
         """
-        rval = []
+        folder_container = []
         current_user_roles = trans.get_current_user_roles()
 
-        def traverse( folder ):
-            admin = trans.user_is_admin()
-            rval = []
-            for subfolder in folder.active_folders:
-                if not admin:
-                    can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
-                if (admin or can_access) and not subfolder.deleted:
-                    subfolder.api_type = 'folder'
-                    rval.append( subfolder )
-            for ld in folder.datasets:
-                if not admin:
-                    can_access = trans.app.security_agent.can_access_dataset( current_user_roles, ld.library_dataset_dataset_association.dataset )
-                if (admin or can_access) and not ld.deleted:
-                    ld.api_type = 'file'
-                    rval.append( ld )
-            return rval
-
-        try:
-            decoded_folder_id = trans.security.decode_id( folder_id[-16:] )
-        except TypeError:
-            trans.response.status = 400
-            return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
+        if ( folder_id.startswith( 'F' ) ):
+            try:
+                decoded_folder_id = trans.security.decode_id( folder_id[1:] )
+            except TypeError:
+                trans.response.status = 400
+                return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
 
         try:
             folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( decoded_folder_id )
-            parent_library = folder.parent_library
         except:
             folder = None
-            log.error( "FolderContentsController.index: Unable to retrieve folder %s"
-                      % folder_id )
+            log.error( "FolderContentsController.index: Unable to retrieve folder with ID: %s" % folder_id )
 
-        # TODO: Find the API's path to this folder if necessary.
-        # This was needed in recursive descent, but it's not needed
-        # for "ls"-style content checking:
-        if not folder or not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+        # We didn't find the folder or user does not have an access to it.
+        if not folder:
             trans.response.status = 400
             return "Invalid folder id ( %s ) specified." % str( folder_id )
+        
+        if not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+            log.warning( "SECURITY: User (id: %s) without proper access rights is trying to load folder with ID of %s" % ( trans.user.id, folder.id ) )
+            trans.response.status = 400
+            return "Invalid folder id ( %s ) specified." % str( folder_id )
+        
+        path_to_root = []
+        def build_path ( folder ):
+            """
+            Search the path upwards recursively and load the whole route of names and ids for breadcrumb purposes.
+            """
+            path_to_root = []
+            # We are almost in root
+            if folder.parent_id is None:
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
+            else:
+            # We add the current folder and traverse up one folder.
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
+                upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_id )
+                path_to_root.extend( build_path( upper_folder ) )
+            return path_to_root
+            
+        # Return the reversed path so it starts with the library node.
+        full_path = build_path( folder )[::-1]
+        folder_container.append( dict( full_path = full_path ) )
+        
+        folder_contents = []
+        time_updated = ''
+        time_created = ''
+        # Go through every item in the folder and include its meta-data.
+        for content_item in self.load_folder_contents( trans, folder ):
+#             rval = content_item.to_dict()
+            return_item = {}
+            encoded_id = trans.security.encode_id( content_item.id )
+            time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+            time_created = content_item.create_time.strftime( "%Y-%m-%d %I:%M %p" )
+#             log.debug('XXXXXXXXXXXXXXXXXXX api type: ' + str(content_item.api_type))
+#             log.debug('XXXXXXXX ALL: ' + str(content_item.__dict__))
+            
+            # For folder return also hierarchy values
+            if content_item.api_type == 'folder':
+                encoded_id = 'F' + encoded_id
+#                 time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+                return_item.update ( dict ( item_count = content_item.item_count ) )
 
-        for content in traverse( folder ):
-            encoded_id = trans.security.encode_id( content.id )
-            if content.api_type == 'folder':
-                encoded_id = 'F' + encoded_id
-            rval.append( dict( id = encoded_id,
-                               type = content.api_type,
-                               name = content.name,
-                               url = url_for( 'folder_contents', folder_id=encoded_id ) ) )
-        return rval
+            if content_item.api_type == 'file':
+#                 log.debug('XXXXX content item class: ' + str(content_item.__class__))
+                library_dataset_dict = content_item.to_dict()
+                library_dataset_dict['data_type']
+                library_dataset_dict['file_size']
+                library_dataset_dict['date_uploaded']
+                return_item.update ( dict ( data_type = library_dataset_dict['data_type'],
+                                            file_size = library_dataset_dict['file_size'],
+                                            date_uploaded = library_dataset_dict['date_uploaded'] ) )
+
+            # For every item return also the default meta-data
+            return_item.update( dict( id = encoded_id,
+                               type = content_item.api_type,
+                               name = content_item.name,
+                               time_updated = time_updated,
+                               time_created = time_created
+                                ) )
+            folder_contents.append( return_item )
+        # Put the data in the container
+        folder_container.append( dict( folder_contents = folder_contents ) )
+        return folder_container
 
     @web.expose_api
     def show( self, trans, id, library_id, **kwd ):
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- /dev/null
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -0,0 +1,241 @@
+"""
+API operations on the dataset from library.
+"""
+import glob
+import logging
+import operator
+import os
+import os.path
+import string
+import sys
+import tarfile
+import tempfile
+import urllib
+import urllib2
+import zipfile
+from galaxy.security import Action
+from galaxy import util, web
+from galaxy.util.streamball import StreamBall
+from galaxy.web.base.controller import BaseAPIController, UsesLibraryMixinItems
+
+import logging
+log = logging.getLogger( __name__ )
+
+# Test for available compression types
+# tmpd = tempfile.mkdtemp()
+# comptypes = []
+# for comptype in ( 'gz', 'bz2' ):
+#     tmpf = os.path.join( tmpd, 'compression_test.tar.' + comptype )
+#     try:
+#         archive = tarfile.open( tmpf, 'w:' + comptype )
+#         archive.close()
+#         comptypes.append( comptype )
+#     except tarfile.CompressionError:
+#         log.exception( "Compression error when testing %s compression.  This option will be disabled for library downloads." % comptype )
+#     try:
+#         os.unlink( tmpf )
+#     except OSError:
+#         pass
+ziptype = '32'
+# tmpf = os.path.join( tmpd, 'compression_test.zip' )
+# try:
+#     archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
+#     archive.close()
+#     comptypes.append( 'zip' )
+#     ziptype = '64'
+# except RuntimeError:
+#     log.exception( "Compression error when testing zip compression. This option will be disabled for library downloads." )
+# except (TypeError, zipfile.LargeZipFile):
+#     # ZIP64 is only in Python2.5+.  Remove TypeError when 2.4 support is dropped
+#     log.warning( 'Max zip file size is 2GB, ZIP64 not supported' )
+#     comptypes.append( 'zip' )
+# try:
+#     os.unlink( tmpf )
+# except OSError:
+#     pass
+# os.rmdir( tmpd )
+
+
+
+class DatasetsController( BaseAPIController, UsesLibraryMixinItems ):
+
+    @web.expose_api
+    def show( self, trans, id, **kwd ):
+        """
+        GET /api/libraries/datasets/{encoded_dataset_id}
+        Displays information about the dataset identified by the lda ID.
+        """
+        # Get dataset.
+        try:
+            dataset = self.get_library_dataset( trans, id = id )
+        except Exception, e:
+            return str( e )
+        try:
+            # Default: return dataset as dict.
+            rval = dataset.to_dict()
+        except Exception, e:
+            rval = "Error in dataset API at listing contents: " + str( e )
+            log.error( rval + ": %s" % str(e), exc_info=True )
+            trans.response.status = 500
+        
+        rval['id'] = trans.security.encode_id(rval['id']);
+        rval['ldda_id'] = trans.security.encode_id(rval['ldda_id']);
+        rval['folder_id'] = 'f' + trans.security.encode_id(rval['folder_id'])
+        
+        return rval
+    
+    @web.expose
+    def download( self, trans, format, **kwd ):
+        """
+        POST /api/libraries/datasets/download/{format}
+        POST data: ldda_ids = []
+        Downloads dataset(s) in the requested format.
+        """
+        lddas = []
+#         is_admin = trans.user_is_admin()
+#         current_user_roles = trans.get_current_user_roles()
+        
+        datasets_to_download = kwd['ldda_ids%5B%5D']
+        
+        if ( datasets_to_download != None ):
+            datasets_to_download = util.listify( datasets_to_download )
+            for dataset_id in datasets_to_download:
+                try:
+                    ldda = trans.sa_session.query( trans.app.model.LibraryDatasetDatasetAssociation ).get( trans.security.decode_id( dataset_id ) )
+                    assert not ldda.dataset.purged
+                    lddas.append( ldda )
+                except:
+                    ldda = None
+                    message += "Invalid library dataset id (%s) specified.  " % str( dataset_id )
+                    
+        if format in [ 'zip','tgz','tbz' ]:
+                error = False
+                killme = string.punctuation + string.whitespace
+                trantab = string.maketrans(killme,'_'*len(killme))
+                try:
+                    outext = 'zip'
+                    if format == 'zip':
+                        # Can't use mkstemp - the file must not exist first
+                        tmpd = tempfile.mkdtemp()
+                        util.umask_fix_perms( tmpd, trans.app.config.umask, 0777, self.app.config.gid )
+                        tmpf = os.path.join( tmpd, 'library_download.' + format )
+                        if ziptype == '64' and trans.app.config.upstream_gzip:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED, True )
+                        elif ziptype == '64':
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
+                        elif trans.app.config.upstream_gzip:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED )
+                        else:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED )
+                        archive.add = lambda x, y: archive.write( x, y.encode('CP437') )
+                    elif format == 'tgz':
+                        if trans.app.config.upstream_gzip:
+                            archive = StreamBall( 'w|' )
+                            outext = 'tar'
+                        else:
+                            archive = StreamBall( 'w|gz' )
+                            outext = 'tgz'
+                    elif format == 'tbz':
+                        archive = StreamBall( 'w|bz2' )
+                        outext = 'tbz2'
+                except ( OSError, zipfile.BadZipfile ):
+                    error = True
+                    log.exception( "Unable to create archive for download" )
+                    message = "Unable to create archive for download, please report this error"
+                    status = 'error'
+                except:
+                     error = True
+                     log.exception( "Unexpected error %s in create archive for download" % sys.exc_info()[0] )
+                     message = "Unable to create archive for download, please report - %s" % sys.exc_info()[0]
+                     status = 'error'
+                if not error:
+                    composite_extensions = trans.app.datatypes_registry.get_composite_extensions()
+                    seen = []
+                    for ldda in lddas:
+                        if ldda.dataset.state in [ 'new', 'upload', 'queued', 'running', 'empty', 'discarded' ]:
+                            continue
+                        ext = ldda.extension
+                        is_composite = ext in composite_extensions
+                        path = ""
+                        parent_folder = ldda.library_dataset.folder
+                        while parent_folder is not None:
+                            # Exclude the now-hidden "root folder"
+                            if parent_folder.parent is None:
+                                path = os.path.join( parent_folder.library_root[0].name, path )
+                                break
+                            path = os.path.join( parent_folder.name, path )
+                            parent_folder = parent_folder.parent
+                        path += ldda.name
+                        while path in seen:
+                            path += '_'
+                        seen.append( path )
+                        zpath = os.path.split(path)[-1] # comes as base_name/fname
+                        outfname,zpathext = os.path.splitext(zpath)
+                        if is_composite:
+                            # need to add all the components from the extra_files_path to the zip
+                            if zpathext == '':
+                                zpath = '%s.html' % zpath # fake the real nature of the html file
+                            try:
+                                archive.add(ldda.dataset.file_name,zpath) # add the primary of a composite set
+                            except IOError:
+                                error = True
+                                log.exception( "Unable to add composite parent %s to temporary library download archive" % ldda.dataset.file_name)
+                                message = "Unable to create archive for download, please report this error"
+                                status = 'error'
+                                continue
+                            flist = glob.glob(os.path.join(ldda.dataset.extra_files_path,'*.*')) # glob returns full paths
+                            for fpath in flist:
+                                efp,fname = os.path.split(fpath)
+                                if fname > '':
+                                    fname = fname.translate(trantab)
+                                try:
+                                    archive.add( fpath,fname )
+                                except IOError:
+                                    error = True
+                                    log.exception( "Unable to add %s to temporary library download archive %s" % (fname,outfname))
+                                    message = "Unable to create archive for download, please report this error"
+                                    status = 'error'
+                                    continue
+                        else: # simple case
+                            try:
+                                archive.add( ldda.dataset.file_name, path )
+                            except IOError:
+                                error = True
+                                log.exception( "Unable to write %s to temporary library download archive" % ldda.dataset.file_name)
+                                message = "Unable to create archive for download, please report this error"
+                                status = 'error'
+                    if not error:
+                        lname = 'selected_dataset'
+                        fname = lname.replace( ' ', '_' ) + '_files'
+                        if format == 'zip':
+                            archive.close()
+                            trans.response.set_content_type( "application/octet-stream" )
+                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                            archive = util.streamball.ZipBall(tmpf, tmpd)
+                            archive.wsgi_status = trans.response.wsgi_status()
+                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            return archive.stream
+                        else:
+                            trans.response.set_content_type( "application/x-tar" )
+                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                            archive.wsgi_status = trans.response.wsgi_status()
+                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            return archive.stream
+        elif format == 'uncompressed':
+            if len(lddas) != 1:
+                return 'Wrong request'
+            else:
+                single_dataset = lddas[0]
+                trans.response.set_content_type( single_dataset.get_mime() )
+                fStat = os.stat( ldda.file_name )
+                trans.response.headers[ 'Content-Length' ] = int( fStat.st_size )
+                valid_chars = '.,^_-()[]0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
+                fname = ldda.name
+                fname = ''.join( c in valid_chars and c or '_' for c in fname )[ 0:150 ]
+                trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s"' % fname
+                try:
+                    return open( single_dataset.file_name )
+                except:
+                    return 'This dataset contains no content'
+        else:
+            return 'Wrong format';
\ No newline at end of file
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 lib/galaxy/webapps/galaxy/api/libraries.py
--- a/lib/galaxy/webapps/galaxy/api/libraries.py
+++ b/lib/galaxy/webapps/galaxy/api/libraries.py
@@ -49,9 +49,10 @@
                            trans.model.Library.table.c.id.in_( accessible_restricted_library_ids ) ) )
         rval = []
         for library in query:
-            item = library.to_dict()
+            item = library.to_dict( view='element' )
             item['url'] = url_for( route, id=trans.security.encode_id( library.id ) )
-            item['id'] = trans.security.encode_id( item['id'] )
+            item['id'] = 'F' + trans.security.encode_id( item['id'] )
+            item['root_folder_id'] = 'F' + trans.security.encode_id( item['root_folder_id'] )
             rval.append( item )
         return rval
 
@@ -131,6 +132,9 @@
         rval['name'] = name
         rval['id'] = encoded_id
         return rval
+    
+    def edit( self, trans, payload, **kwd  ):
+        return "Not implemented yet"
 
     @web.expose_api
     def delete( self, trans, id, **kwd ):
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -46,6 +46,7 @@
     atexit.register( app.shutdown )
     # Create the universe WSGI application
     webapp = GalaxyWebApplication( app, session_cookie='galaxysession', name='galaxy' )
+    # Add the controllers folder
     # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
     webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
                       controller='admin_toolshed',
@@ -53,11 +54,17 @@
                       repository_id=None,
                       image_file=None )
     webapp.add_ui_controllers( 'galaxy.webapps.galaxy.controllers', app )
+    # Add the api folder
+    webapp.add_api_controllers( 'galaxy.webapps.galaxy.api', app )
+    # Add the api folder VERSION 2
+    webapp.add_api_controllers( 'galaxy.webapps.galaxy.api.v2', app )
+
     # Force /history to go to /root/history -- needed since the tests assume this
     webapp.add_route( '/history', controller='root', action='history' )
     # Force /activate to go to the controller
     webapp.add_route( '/activate', controller='user', action='activate' )
     # These two routes handle our simple needs at the moment
+    
     webapp.add_route( '/async/:tool_id/:data_id/:data_secret', controller='async', action='index', tool_id=None, data_id=None, data_secret=None )
     webapp.add_route( '/:controller/:action', action='index' )
     webapp.add_route( '/:action', controller='root', action='index' )
@@ -66,8 +73,6 @@
     webapp.add_route( '/datasets/:dataset_id/display/{filename:.+?}', controller='dataset', action='display', dataset_id=None, filename=None)
     webapp.add_route( '/datasets/:dataset_id/:action/:filename', controller='dataset', action='index', dataset_id=None, filename=None)
     webapp.add_route( '/display_application/:dataset_id/:app_name/:link_name/:user_id/:app_action/:action_param',
-        controller='dataset', action='display_application', dataset_id=None, user_id=None,
-        app_name = None, link_name = None, app_action = None, action_param = None )
     webapp.add_route( '/u/:username/d/:slug/:filename', controller='dataset', action='display_by_username_and_slug', filename=None )
     webapp.add_route( '/u/:username/p/:slug', controller='page', action='display_by_username_and_slug' )
     webapp.add_route( '/u/:username/h/:slug', controller='history', action='display_by_username_and_slug' )
@@ -171,9 +176,44 @@
     #webapp.mapper.connect( 'run_workflow', '/api/workflow/{workflow_id}/library/{library_id}', controller='workflows', action='run', workflow_id=None, library_id=None, conditions=dict(method=["GET"]) )
     webapp.mapper.resource( 'search', 'search', path_prefix='/api' )
 
+    webapp.mapper.resource_with_deleted( 'quota', 'quotas', path_prefix='/api' )
+    webapp.mapper.resource_with_deleted( 'history', 'histories', path_prefix='/api' )
+    webapp.mapper.resource_with_deleted( 'user', 'users', path_prefix='/api' )
+
+    # =======================
+    # ===== LIBRARY API =====
+    # =======================
+
+    webapp.mapper.connect( 'show_lda_item', '/api/libraries/datasets/:id', controller='lda_datasets', action='show', conditions=dict(method=["GET"]) )
+    webapp.mapper.connect( 'download_lda_items', '/api/libraries/datasets/download/:format', controller='lda_datasets', action='download', conditions=dict(method=["POST"]) )
+
+    webapp.mapper.resource_with_deleted( 'library', 'libraries', path_prefix='/api' )
+    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
+
+    webapp.mapper.resource( 'content', 'contents',
+                                controller='folder_contents',
+                                name_prefix='folder_',
+                                path_prefix='/api/folders/:folder_id',
+                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
+    webapp.mapper.resource( 'content',
+                                'contents',
+                                controller='library_contents',
+                                name_prefix='library_',
+                                path_prefix='/api/libraries/:library_id',
+                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    webapp.mapper.resource( 'permission',
+                                'permissions',
+                                path_prefix='/api/libraries/:library_id',
+                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    _add_item_extended_metadata_controller( webapp,
+                               name_prefix="library_dataset_",
+                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
+    
+
     # add as a non-ATOM API call to support the notion of a 'current/working' history unique to the history resource
     webapp.mapper.connect( "set_as_current", "/api/histories/{id}/set_as_current",
         controller="histories", action="set_as_current", conditions=dict( method=["POST"] ) )
+    webapp.mapper.connect( "set_as_current", "/api/histories/{id}/set_as_current", controller="histories", action="set_as_current", conditions=dict( method=["POST"] ) )
 
     webapp.mapper.connect( "create_api_key", "/api/users/:user_id/api_key",
         controller="users", action="api_key", user_id=None, conditions=dict( method=["POST"] ) )
@@ -181,6 +221,7 @@
     # visualizations registry generic template renderer
     webapp.add_route( '/visualization/show/:visualization_name',
         controller='visualization', action='render', visualization_name=None )
+    webapp.add_route( '/visualization/show/:visualization_name', controller='visualization', action='render', visualization_name=None )
 
     # "POST /api/workflows/import"  =>  ``workflows.import_workflow()``.
     # Defines a named route "import_workflow".
@@ -188,6 +229,15 @@
     webapp.mapper.connect("workflow_dict", '/api/workflows/{workflow_id}/download', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
     # Preserve the following download route for now for dependent applications  -- deprecate at some point
     webapp.mapper.connect("workflow_dict", '/api/workflows/download/{workflow_id}', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
+    
+    
+    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
+    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
+                      controller='admin_toolshed',
+                      action='display_image_in_repository',
+                      repository_id=None,
+                      image_file=None )
+
     # Galaxy API for tool shed features.
     webapp.mapper.resource( 'tool_shed_repository',
                             'tool_shed_repositories',
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 lib/galaxy/webapps/galaxy/controllers/library.py
--- a/lib/galaxy/webapps/galaxy/controllers/library.py
+++ b/lib/galaxy/webapps/galaxy/controllers/library.py
@@ -76,6 +76,17 @@
 
     library_list_grid = LibraryListGrid()
 
+    
+    @web.expose
+    def list( self, trans, **kwd ):
+        params = util.Params( kwd )
+        # define app configuration for generic mako template
+        app = {
+            'jscript'       : "galaxy.library"
+        }
+        # fill template
+        return trans.fill_template('galaxy.panels.mako', config = {'app' : app})
+
     @web.expose
     def index( self, trans, **kwd ):
         params = util.Params( kwd )
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 static/scripts/galaxy.library.js
--- /dev/null
+++ b/static/scripts/galaxy.library.js
@@ -0,0 +1,860 @@
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+// === GALAXY LIBRARY MODULE ====
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+
+// global variables
+var view            = null;
+var library_router  = null;
+var responses = [];
+
+
+// load required libraries
+// require([
+//     // load js libraries
+//     'utils/galaxy.css',
+//     ], function(css){
+//         // load css
+//         css.load_file("static/style/library.css");
+// });
+
+// dependencies
+define(["galaxy.modal", "galaxy.masthead", "utils/galaxy.utils"], function(mod_modal, mod_masthead, mod_utils) {
+
+// MMMMMMMMMMMMMMM
+// === Models ====
+// MMMMMMMMMMMMMMM
+
+    // LIBRARY
+    var Library = Backbone.Model.extend({
+      urlRoot: '/api/libraries'
+  });
+
+    // LIBRARIES
+    var Libraries = Backbone.Collection.extend({
+      url: '/api/libraries',
+      model: Library
+  });
+
+    // ITEM
+    var Item = Backbone.Model.extend({
+        urlRoot : '/api/libraries/datasets'
+    })
+
+    // FOLDER
+    var Folder = Backbone.Collection.extend({
+      model: Item      
+  })
+
+    // CONTAINER for folder contents (folders, items and metadata).
+    var FolderContainer = Backbone.Model.extend({
+     defaults : {
+        folder : new Folder(),
+        full_path : "unknown",
+        urlRoot : "/api/folders/",
+        id : "unknown"
+    },
+    parse : function(obj) {
+      this.full_path = obj[0].full_path;
+          // update the inner collection
+          this.get("folder").reset(obj[1].folder_contents);
+          return obj;
+      }
+  })
+
+    // HISTORY ITEM
+    var HistoryItem = Backbone.Model.extend({
+        urlRoot : '/api/histories/'
+    });
+
+    // HISTORY
+    var GalaxyHistory = Backbone.Model.extend({
+        url : '/api/histories/'
+    });
+
+    // HISTORIES
+    var GalaxyHistories = Backbone.Collection.extend({
+        url : '/api/histories',
+        model : GalaxyHistory
+    });
+
+    //ROUTER
+    var LibraryRouter = Backbone.Router.extend({    
+        routes: {
+            "" :                 "libraries",
+            "folders/:id" :      "folder_content",
+            "folders/:folder_id/download/:format" : "download"
+        }
+    });
+
+
+// MMMMMMMMMMMMMM
+// === VIEWS ====
+// MMMMMMMMMMMMMM
+
+// galaxy folder
+var FolderContentView = Backbone.View.extend({
+    // main element definition
+    el : '#center',
+    // progress percentage
+    progress: 0,
+    // progress rate per one item
+    progressStep: 1, 
+    // last selected history in modal for UX
+    lastSelectedHistory: '',
+    // self modal
+    modal : null,
+    // loaded folders
+    folders : null,
+    
+    // initialize
+    initialize : function(){
+        this.folders = [];
+        this.queue = jQuery.Deferred();
+        this.queue.resolve();
+    },    
+
+// MMMMMMMMMMMMMMMMMM
+// === TEMPLATES ====
+// MMMMMMMMMMMMMMMMMM
+
+    // set up 
+    templateFolder : function (){
+        var tmpl_array = [];
+
+        // CONTAINER
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; ">');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
+
+        // TOOLBAR
+        tmpl_array.push('<div id="library_folder_toolbar" >');
+        tmpl_array.push('   <button title="Create New Folder" id="toolbtn_create_folder" class="btn btn-primary" type="button"><span class="fa fa-icon-plus"></span><span class="fa fa-icon-folder-close"></span> folder</button>');
+        tmpl_array.push('   <button id="toolbtn_bulk_import" class="btn btn-primary" style="display: none; margin-left: 0.5em;" type="button"><span class="fa fa-icon-external-link"></span> to history</button>');
+        
+        tmpl_array.push('   <div id="toolbtn_dl" class="btn-group" style="margin-left: 0.5em; display: none; ">');
+        tmpl_array.push('       <button id="drop_toggle" type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown">');
+        tmpl_array.push('       <span class="fa fa-icon-download"></span> download <span class="caret"></span>');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       <ul class="dropdown-menu" role="menu">');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tgz">.tar.gz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tbz">.tar.bz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/zip">.zip</a></li>');
+        tmpl_array.push('       </ul>');
+        tmpl_array.push('   </div>');
+
+        tmpl_array.push('</div>');
+
+        // BREADCRUMBS
+        tmpl_array.push('<div class="library_breadcrumb">');
+        tmpl_array.push('<a title="Return to the list of libraries" href="#">Libraries</a><b>|</b> ');
+        tmpl_array.push('<% _.each(path, function(path_item) { %>');
+        tmpl_array.push('<% if (path_item[0] != id) { %>');
+        tmpl_array.push('<a title="Return to this folder" href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a><b>|</b> ');
+        tmpl_array.push('<% } else { %>');
+        tmpl_array.push('<span title="You are in this folder"><%- path_item[1] %></span>');
+        tmpl_array.push('<% } %>');
+        tmpl_array.push('<% }); %>');
+        tmpl_array.push('</div>');
+
+        // FOLDER CONTENT
+        tmpl_array.push('<table id="folder_table" class="table table-condensed">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
+        tmpl_array.push('       <th class="button_heading">view</th>');
+        tmpl_array.push('       <th>name</th>');
+        tmpl_array.push('       <th>data type</th>');
+        tmpl_array.push('       <th>size</th>');
+        tmpl_array.push('       <th>date</th>');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td><button title="Go to parent folder" type="button" data-id="<%- upper_folder_id %>" class="btn_open_folder btn btn-default btn-xs">');        
+        tmpl_array.push('       <span class="fa fa-icon-arrow-up"></span> .. go up</td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       </tr>');        
+        tmpl_array.push('       <% _.each(items, function(content_item) { %>');
+        tmpl_array.push('       <tr class="folder_row light" id="<%- content_item.id %>">');
+        tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>'); // folder
+        tmpl_array.push('               <td></td>');
+        tmpl_array.push('               <td><button title="Open this folder" type="button" data-id="<%- content_item.id %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- content_item.get("name") %>');
+        tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>'); // empty folder
+        tmpl_array.push('           <span class="muted">(empty folder)</span>');
+        tmpl_array.push('           <% } %>');
+        tmpl_array.push('           </td>');
+        tmpl_array.push('           <td>folder</td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("item_count")) %> item(s)</td>'); // size
+        tmpl_array.push('           <% } else {  %>');
+        tmpl_array.push('           <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
+        tmpl_array.push('       <td>');
+        tmpl_array.push('       <button title="See details of this dataset" type="button" class="library-dataset btn btn-default btn-xs">');
+        tmpl_array.push('       <span class="fa fa-icon-eye-open"></span> details');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       </td>');
+        tmpl_array.push('           <td><%- content_item.get("name") %></td>'); // dataset
+        tmpl_array.push('           <td><%= _.escape(content_item.get("data_type")) %></td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("readable_size")) %></td>'); // size
+        tmpl_array.push('           <% } %>  ');
+        tmpl_array.push('           <td><%= _.escape(content_item.get("time_updated")) %></td>'); // time updated
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('       ');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+
+        tmpl_array.push('</div>');
+        return tmpl_array.join('');
+    },
+    templateDatasetModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<div id="dataset_info_modal">');
+        tmpl_array.push('   <table class="table table-striped table-condensed">');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row" id="id_row" data-id="<%= _.escape(item.get("ldda_id")) %>">Name</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("name")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Data type</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("data_type")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Genome build</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("genome_build")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('           <th scope="row">Size</th>');
+        tmpl_array.push('           <td><%= _.escape(size) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Date uploaded</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("date_uploaded")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Uploaded by</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("uploaded_by")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('           <tr scope="row">');
+        tmpl_array.push('           <th scope="row">Data Lines</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_data_lines")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <th scope="row">Comment Lines</th>');
+        tmpl_array.push('           <% if (item.get("metadata_comment_lines") === "") { %>'); //folder        
+        tmpl_array.push('               <td scope="row"><%= _.escape(item.get("metadata_comment_lines")) %></td>');
+        tmpl_array.push('           <% } else { %>'); 
+        tmpl_array.push('               <td scope="row">unknown</td>');    
+        tmpl_array.push('           <% } %>'); 
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Number of Columns</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_columns")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Column Types</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_column_types")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Miscellaneous information</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("misc_blurb")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('   </table>');
+        tmpl_array.push('   <pre class="peek">');
+        tmpl_array.push('   </pre>');
+        tmpl_array.push('</div>');
+
+        return tmpl_array.join('');
+    },
+
+    templateHistorySelectInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_combo" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_single" name="dataset_import_single" style="width:50%; margin-bottom: 1em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
+
+        return tmpl_array.join('');
+        },   
+
+    templateBulkImportInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_combo_bulk" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_bulk" name="dataset_import_bulk" style="width:50%; margin-bottom: 1em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
+
+        return tmpl_array.join('');
+        },    
+
+      // convert size to nice string
+      size_to_string : function (size)
+      {
+            // identify unit
+            var unit = "";
+            if (size >= 100000000000)   { size = size / 100000000000; unit = "TB"; } else
+            if (size >= 100000000)      { size = size / 100000000; unit = "GB"; } else
+            if (size >= 100000)         { size = size / 100000; unit = "MB"; } else
+            if (size >= 100)            { size = size / 100; unit = "KB"; } else
+            { size = size * 10; unit = "b"; }
+            // return formatted string
+            return (Math.round(size) / 10) + unit;
+        },
+
+// MMMMMMMMMMMMMMM
+// === EVENTS ====
+// MMMMMMMMMMMMMMM
+
+      // event binding
+      events: {
+            'click #select-all-checkboxes' : 'selectAll',
+            'click .folder_row' : 'selectClicked',
+            'click #toolbtn_bulk_import' : 'modalBulkImport',
+            'click #toolbtn_dl' : 'bulkDownload',
+            'click .library-dataset' : 'showDatasetDetails',
+            'click #toolbtn_create_folder' : 'createFolderModal',
+            'click .btn_open_folder' : 'navigateToFolder'
+        },
+
+      //render the folder view
+      render: function (options) {
+        //hack to show scrollbars
+        $("#center").css('overflow','auto');
+
+        view = this;
+        var that = this;
+
+        var folderContainer = new FolderContainer({id: options.id});
+        folderContainer.url = folderContainer.attributes.urlRoot + options.id + '/contents';
+
+        folderContainer.fetch({
+          success: function (container) {
+
+            // prepare nice size strings
+              for (var i = 0; i < folderContainer.attributes.folder.models.length; i++) {
+                  var model = folderContainer.attributes.folder.models[i]
+                  if (model.get('type') === 'file'){
+                    model.set('readable_size', that.size_to_string(model.get('file_size')))
+                  }
+              };
+
+              // find the upper id
+              var path = folderContainer.full_path;
+              var upper_folder_id;
+              if (path.length === 1){ // library is above us
+                upper_folder_id = 0;
+              } else {
+                upper_folder_id = path[path.length-2][0];
+              }
+
+              var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id, upper_folder_id: upper_folder_id });
+              // var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id });
+              that.$el.html(template);
+            }
+          })
+        },
+
+      // handles the click on 'open' and 'upper' folder icons
+      navigateToFolder : function(event){
+        var folder_id = $(event.target).attr('data-id');
+        if (typeof folder_id === 'undefined') {
+            return false;
+        } else if (folder_id === '0'){
+            library_router.navigate('#', {trigger: true, replace: true});
+        } else {
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+        }
+      },
+
+      //show modal with current dataset info
+      showDatasetDetails : function(event){
+        // prevent default
+        event.preventDefault();
+        
+//TODO check whether we already have the data
+
+        //load the ID of the row
+        var id = $(event.target).parent().parent().attr('id');
+        
+        //create new item
+        var item = new Item();
+        var histories = new GalaxyHistories();
+        item.id = id;
+        var self = this;
+
+        //fetch the dataset info
+        item.fetch({
+          success: function (item) {
+// TODO can render here already
+                //fetch user histories for import purposes
+                histories.fetch({
+                    success: function (histories){self.renderModalAfterFetch(item, histories)}
+                });
+            }
+        });
+    },
+
+      // show the current dataset in a modal
+      renderModalAfterFetch : function(item, histories){
+        var size = this.size_to_string(item.get('file_size'));
+        var template = _.template(this.templateDatasetModal(), { item : item, size : size });
+        this.modal = null;
+            // make modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal({
+                title   : 'Dataset Details',
+                body    : template,
+                buttons : {
+                    'Import' : function() { self.importCurrentIntoHistory() },
+                    'Download' : function() { self.downloadCurrent() },
+                    'Close'  : function() { self.modal.hide(); $('.modal').remove(); self.modal = null; } // TODO refill nicely modal with data
+                }
+            });
+            $(".peek").html(item.get("peek"));
+            var history_footer_tmpl = _.template(this.templateHistorySelectInModal(), {histories : histories.models});
+            $(this.modal.elMain).find('.buttons').prepend(history_footer_tmpl);
+
+            // preset last selected history if we know it
+            if (self.lastSelectedHistory.length > 0) {
+                $(this.modal.elMain).find('#dataset_import_single').val(self.lastSelectedHistory);
+            }
+
+            // show the prepared modal
+            this.modal.show();
+        },
+
+        // download dataset shown currently in modal
+        downloadCurrent : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
+
+            var library_dataset_id = [];
+            library_dataset_id.push($('#id_row').attr('data-id'));
+            var url = '/api/libraries/datasets/download/uncompressed';
+            var data = {'ldda_ids' : library_dataset_id};
+
+            // we assume the view is existent
+            folderContentView.processDownload(url, data);
+            this.modal.enableButton('Import');
+            this.modal.enableButton('Download');
+        },
+
+        // import dataset shown currently in modal into selected history
+        importCurrentIntoHistory : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
+
+            var history_id = $(this.modal.elMain).find('select[name=dataset_import_single] option:selected').val();
+            this.lastSelectedHistory = history_id; //save selected history for further use
+
+            var library_dataset_id = $('#id_row').attr('data-id');
+            var historyItem = new HistoryItem();
+            var self = this;
+            historyItem.url = historyItem.urlRoot + history_id + '/contents';
+
+            // save the dataset into selected history
+            historyItem.save({ content : library_dataset_id, source : 'library' }, { success : function(){
+                self.modal.showNotification('Dataset imported', 3000, 'success');
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');
+            }, error : function(){
+                self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, 'error');
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');                
+            }
+        });
+        },
+
+        // select all datasets 
+        selectAll : function (event) {
+           var selected = event.target.checked;
+                 // Iterate each checkbox
+                 $(':checkbox').each(function () { this.checked = selected; });
+                 this.showTools();
+             },
+
+         // click checkbox on folder click
+         selectClicked : function (event) {
+            var checkbox = $("#" + event.target.parentElement.id).find(':checkbox')
+                if (checkbox[0] != undefined) {
+                  if (checkbox[0].checked){
+                    checkbox[0].checked = '';
+                    // $(event.target.parentElement).css('background-color', '').css('color', '');
+                    $(event.target.parentElement).removeClass('dark');
+                    $(event.target.parentElement).find('a').removeClass('dark');
+                    $(event.target.parentElement).addClass('light');
+                    $(event.target.parentElement).find('a').addClass('light');
+                } else {
+                    checkbox[0].checked = 'selected';
+                    $(event.target.parentElement).removeClass('light');
+                    $(event.target.parentElement).find('a').removeClass('light');
+                    $(event.target.parentElement).addClass('dark');
+                    $(event.target.parentElement).find('a').addClass('dark');
+                    // $(event.target.parentElement).css('background-color', '#8389a1').css('color', 'white');
+                }
+            }
+            this.showTools();
+        },
+
+        // show toolbar in case something is selected
+        showTools : function(){
+            var checkedValues = $('#folder_table').find(':checked');
+            if(checkedValues.length > 0){
+                $('#toolbtn_bulk_import').show();
+                $('#toolbtn_dl').show();
+            } else {
+                $('#toolbtn_bulk_import').hide();
+                $('#toolbtn_dl').hide();
+            }
+
+        },
+
+        // show bulk import modal
+        modalBulkImport : function(){
+            var self = this;
+            // fetch histories
+            var histories = new GalaxyHistories();
+            histories.fetch({
+                    success: function (histories){
+                        // make modal
+                        var history_modal_tmpl =  _.template(self.templateBulkImportInModal(), {histories : histories.models});
+                        self.modal = new mod_modal.GalaxyModal({
+                            title   : 'Import into History',
+                            body    : history_modal_tmpl,
+                            buttons : {
+                                'Import' : function() {self.importAllIntoHistory()},
+                                'Close'  : function() {self.modal.hide(); $('.modal').remove(); self.modal = null;}
+                            }
+                        });
+                        // show the prepared modal
+                        self.modal.show();
+                    }
+                });
+        },
+
+        // import all selected datasets into history
+        importAllIntoHistory : function (){
+            //disable the button
+            this.modal.disableButton('Import');
+
+            var history_id = $("select[name=dataset_import_bulk] option:selected").val();
+            var history_name = $("select[name=dataset_import_bulk] option:selected").text();
+
+            var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+            var progress_bar_tmpl = _.template(this.templateProgressBar(), { history_name : history_name });
+            $(this.modal.elMain).find('.modal-body').html(progress_bar_tmpl);
+            
+            // init the progress bar
+            var progressStep = 100 / dataset_ids.length;
+            this.initProgress(progressStep);
+
+            // prepare the dataset objects to be imported
+            var datasets_to_import = [];
+            for (var i = dataset_ids.length - 1; i >= 0; i--) {
+                library_dataset_id = dataset_ids[i];
+                var historyItem = new HistoryItem();
+                var self = this;
+                historyItem.url = historyItem.urlRoot + history_id + '/contents';
+                historyItem.content = library_dataset_id;
+                historyItem.source = 'library';
+                datasets_to_import.push(historyItem);
+            };
+
+            // call the recursive function to call ajax one after each other
+            this.chainCall(datasets_to_import);
+        },
+
+        chainCall: function(history_item_set){
+            var self = this;
+            var popped_item = history_item_set.pop();
+            if (typeof popped_item === "undefined") {
+                self.modal.showNotification('All datasets imported', 3000, 'success');
+                // enable button again
+                self.modal.enableButton('Import');
+                return
+            }
+                var promise = $.when(popped_item.save({content: popped_item.content, source: popped_item.source})).done(function(a1){
+                        self.updateProgress();
+                        responses.push(a1);
+                        self.chainCall(history_item_set);
+                    });
+        },
+
+        initProgress: function(progressStep){
+            this.progress = 0;
+            this.progressStep = progressStep;
+        },
+        updateProgress: function(){
+            this.progress += this.progressStep;
+            $('.progress-bar').width(Math.round(this.progress) + '%');
+            txt_representation = Math.round(this.progress) + '% Complete';
+            $('.completion_span').text(txt_representation);
+        },
+
+      // progress bar 
+      templateProgressBar : function (){
+        var tmpl_array = [];
+
+        tmpl_array.push('<div class="import_text">');
+        tmpl_array.push('Importing selected datasets to history <b><%= _.escape(history_name) %></b>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('<div class="progress">');
+        tmpl_array.push('   <div class="progress-bar" role="progressbar" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100" style="width: 00%;">');
+        tmpl_array.push('       <span class="completion_span">0% Complete</span>');
+        tmpl_array.push('   </div>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('');
+
+        return tmpl_array.join('');
+      },
+
+      // download selected datasets
+      download : function(folder_id, format){
+        var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+
+        var url = '/api/libraries/datasets/download/' + format;
+        var data = {'ldda_ids' : dataset_ids};
+        this.processDownload(url, data);
+      },
+
+      // create hidden form and submit through POST to initialize download
+      processDownload: function(url, data, method){
+        //url and data options required
+        if( url && data ){ 
+                //data can be string of parameters or array/object
+                data = typeof data == 'string' ? data : $.param(data);
+                //split params into form inputs
+                var inputs = '';
+                $.each(data.split('&'), function(){ 
+                        var pair = this.split('=');
+                        inputs+='<input type="hidden" name="'+ pair[0] +'" value="'+ pair[1] +'" />'; 
+                });
+                //send request
+                $('<form action="'+ url +'" method="'+ (method||'post') +'">'+inputs+'</form>')
+                .appendTo('body').submit().remove();
+        };
+      },
+
+      // shows modal for creating folder
+      createFolderModal: function(){
+        alert('creating folder');
+      }
+
+    });
+
+// galaxy library view
+var GalaxyLibraryview = Backbone.View.extend({
+    el: '#center',
+
+    events: {
+        'click #create_new_library_btn' : 'show_library_modal'
+    },
+
+    // initialize
+    initialize : function(){
+    },    
+
+    // template
+    template_library_list : function (){
+        tmpl_array = [];
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; overflow: auto !important; ">');
+
+        tmpl_array.push('');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
+        tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary icon-file ">New Library</a>');
+        tmpl_array.push('<table class="table table-condensed">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('     <th class="button_heading"></th>');
+        tmpl_array.push('     <th>name</th>');
+        tmpl_array.push('     <th>description</th>');
+        tmpl_array.push('     <th>synopsis</th> ');
+        tmpl_array.push('     <th>model type</th> ');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <% _.each(libraries, function(library) { %>');
+        tmpl_array.push('           <tr>');
+        tmpl_array.push('               <td><button title="Open this library" type="button" data-id="<%- library.get("root_folder_id") %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- library.get("name") %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
+        tmpl_array.push('           </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+        
+        tmpl_array.push('</div>');
+        return tmpl_array.join('');
+    },
+
+    // render
+    render: function () {
+        //hack to show scrollbars
+        $("#center").css('overflow','auto');
+        var that = this;
+        // if (typeof libraries === "undefined") {
+          libraries = new Libraries();
+        // }
+        libraries.fetch({
+          success: function (libraries) {
+            var template = _.template(that.template_library_list(), { libraries : libraries.models });
+            that.$el.html(template);
+        }
+    })
+    },
+
+    // own modal
+    modal : null,
+
+    // show/hide create library modal
+    show_library_modal : function (event){
+        event.preventDefault();
+        event.stopPropagation();
+        
+        // create modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal(
+            {
+                title   : 'Create New Library',
+                body    : this.template_new_library(),
+                buttons : {
+                    'Create' : function() {self.create_new_library_event()},
+                    'Close'  : function() {self.modal.hide()}
+                }
+            });
+        
+        // show prepared modal
+        this.modal.show();
+    },
+
+    // create the new library from modal
+    create_new_library_event: function(){
+        var libraryDetails = this.serialize_new_library();
+        var valid = this.validate_new_library(libraryDetails);
+        var library = new Library();
+        var self = this;
+        library.save(libraryDetails, {
+          success: function (library) {
+            self.modal.hide();
+            self.clear_library_modal();
+            self.render();
+          },
+          error: function(){
+            self.modal.showNotification('An error occured', 5000, 'error');
+            }
+    });
+        return false;
+    },
+
+    // clear the library modal once saved
+    clear_library_modal : function(){
+        $("input[name='Name']").val('');
+        $("input[name='Description']").val('');
+        $("input[name='Synopsis']").val('');
+    },
+
+    // serialize data from the form
+    serialize_new_library : function(){
+        return {
+            name: $("input[name='Name']").val(),
+            description: $("input[name='Description']").val(),
+            synopsis: $("input[name='Synopsis']").val()
+        };
+    },
+
+    validate_new_library: function(library){
+
+    },
+
+
+    // template for new library modal
+    template_new_library: function()
+    {
+        tmpl_array = [];
+
+        tmpl_array.push('<div id="new_library_modal">');
+        tmpl_array.push('<form>');
+        tmpl_array.push('<input type="text" name="Name" value="" placeholder="Name">');
+        tmpl_array.push('<input type="text" name="Description" value="" placeholder="Description">');
+        tmpl_array.push('<input type="text" name="Synopsis" value="" placeholder="Synopsis">');
+        tmpl_array.push('</form>');
+        tmpl_array.push('</div>');
+
+        return tmpl_array.join('');
+    }
+});
+
+// galaxy library wrapper View
+var GalaxyLibrary = Backbone.View.extend({
+    folderContentView : null,
+    galaxyLibraryview : null,
+    initialize : function(){
+
+        folderContentView = new FolderContentView();
+        galaxyLibraryview = new GalaxyLibraryview();
+
+        library_router = new LibraryRouter();
+
+        library_router.on('route:libraries', function() {
+          // render libraries list
+          galaxyLibraryview.render();
+      });
+
+        library_router.on('route:folder_content', function(id) {
+          // render folder's contents
+          folderContentView.render({id: id});
+      });    
+
+       library_router.on('route:download', function(folder_id, format) {
+          // send download stream
+          if (typeof folderContentView === 'undefined'){
+            alert('you cant touch this!');
+          // } else if (folderContentView.modal !== null){
+            // folderContentView.download(folder_id, format);
+          } else if ($('#center').find(':checked').length === 0) { // coming from outside of the library app
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+            // TODO show message of redirection
+          } else {
+            folderContentView.download(folder_id, format);
+            library_router.navigate('folders/' + folder_id, {trigger: false, replace: true});
+          }
+
+      });      
+
+Backbone.history.start();   
+
+return this
+}
+});
+
+// return
+return {
+    GalaxyApp: GalaxyLibrary
+};
+
+});
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 static/scripts/galaxy.library.router.js
--- /dev/null
+++ b/static/scripts/galaxy.library.router.js
@@ -0,0 +1,21 @@
+define( [], function() {
+
+/**
+ * -- Routers --
+ */
+
+/**
+ * Router for library browser.
+ */
+var LibraryRouter = Backbone.Router.extend({    
+    routes: {
+        "" :                 "libraries",
+        "folders/:id" :      "folder_content"
+    }
+});
+
+return {
+    LibraryRouter: LibraryRouter,
+};
+
+})
\ No newline at end of file
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 static/scripts/galaxy.menu.js
--- a/static/scripts/galaxy.menu.js
+++ b/static/scripts/galaxy.menu.js
@@ -58,6 +58,12 @@
         tab_shared.add({
             title   : "Data Libraries",
             content : "library/index",
+            content : "library/index"
+        });
+
+        tab_shared.add({
+            title   : "New Libraries",
+            content : "library/list",
             divider : true
         });
 
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -1,6 +1,3 @@
-/*
-    galaxy modal
-*/
 
 // dependencies
 define([], function() {
@@ -25,8 +22,51 @@
     
     // initialize
     initialize : function(options) {
+        self = this;
         if (options)
             this.create(options);
+
+        this.bindClick(event, self);
+    },
+
+    // bind the click-to-hide function
+    bindClick: function(event, that) {
+        // bind the ESC key to hide() function
+        $(document).on('keyup', function(event){
+            if (event.keyCode == 27) { self.hide(); }
+        })
+        // bind the click anywhere to hide() function...
+        $('html').on('click', function(event){
+            that.hide();
+        })
+        // ...but don't hide if the click is on modal content
+        $('.modal-content').on('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+    // unbind the click-to-hide function
+    unbindClick: function(event, that){
+        // bind the ESC key to hide() function
+        $(document).off('keyup', function(event){
+            if (event.keyCode == 27) { that.hide(); }
+        })
+        // unbind the click anywhere to hide() function...
+        $('html').off('click', function(event){
+            that.hide();
+        })
+        // ...but don't hide if the click is on modal content
+        $('.modal-content').off('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+
+    // destroy
+    destroy : function(){
+        this.hide();
+        this.unbindClick();
+        $('.modal').remove();
     },
 
     // adds and displays a new frame/window
@@ -87,6 +127,7 @@
         this.$footer  = (this.$el).find('.modal-footer');
         this.$buttons = (this.$el).find('.buttons');
         this.$backdrop = (this.$el).find('.modal-backdrop');
+        this.$notification = (this.$el).find('.notification-modal');
         
         // append body
         this.$body.html(this.options.body);
@@ -120,6 +161,47 @@
         this.$buttons.find('#' + String(name).toLowerCase()).prop('disabled', true);
     },
     
+    // hide buttons
+    hideButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).hide();
+    },
+    // show buttons
+    showButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).show();
+    },
+
+    // show notification
+    showNotification : function(message, duration, type) {
+        // defaults
+        var duration = typeof duration !== 'undefined' ? duration : 1500;
+        // var bgColor = typeof bgColor !== 'undefined' ? bgColor : "#F4E0E1";
+        // var txtColor = typeof txtColor !== 'undefined' ? txtColor : "#A42732";
+        var bgColor;
+        var txtColor;
+
+        if (type === 'error'){
+            bgColor = '#f4e0e1';
+            txtColor = '#a42732';
+        // } else if (type === 'success'){
+        } else { // success is default
+            bgColor = '#e1f4e0';
+            txtColor = '#32a427'; 
+        }
+
+        var HTMLmessage = "<div class='notification-message' style='text-align:center; line-height:16px; '> " + message + " </div>";
+        this.$notification.html("<div id='notification-bar' style='display:none; float: right; height: 16px; width:100%; background-color: " + bgColor + "; z-index: 100; color: " + txtColor + ";border-bottom: 1px solid " + txtColor + ";'>" + HTMLmessage + "</div>");
+
+        var self = this;
+        
+        /*animate the bar*/
+        $('#notification-bar').slideDown(function() {
+            setTimeout(function() {
+                $('#notification-bar').slideUp(function() {self.$notification.html('');});
+            }, duration);
+        });
+
+    },
+
     // returns scroll top for body element
     scrollTop: function()
     {
@@ -135,10 +217,10 @@
         return  '<div class="modal">' +
                     '<div class="modal-backdrop fade in" style="z-index: -1;"></div>' +
                     '<div class="modal-dialog">' +
-                        '<div class="modal-content">' +
                             '<div class="modal-header">' +
                                 '<button type="button" class="close" style="display: none;">×</button>' +
                                 '<h4 class="title">' + title + '</h4>' +
+                                '<span class="notification-modal"></span>' + 
                             '</div>' +
                             '<div class="modal-body"></div>' +
                             '<div class="modal-footer">' +
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 static/scripts/libs/bootstrap.js
--- a/static/scripts/libs/bootstrap.js
+++ b/static/scripts/libs/bootstrap.js
@@ -575,3 +575,158 @@
   }
 
 }(window.jQuery);
+
+/* ========================================================================
+ * Bootstrap: dropdown.js v3.0.0
+ * http://twbs.github.com/bootstrap/javascript.html#dropdowns
+ * ========================================================================
+ * Copyright 2012 Twitter, Inc.
+ *
+ * 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.
+ * ======================================================================== */
+
+
++function ($) { "use strict";
+
+  // DROPDOWN CLASS DEFINITION
+  // =========================
+
+  var backdrop = '.dropdown-backdrop'
+  var toggle   = '[data-toggle=dropdown]'
+  var Dropdown = function (element) {
+    var $el = $(element).on('click.bs.dropdown', this.toggle)
+  }
+
+  Dropdown.prototype.toggle = function (e) {
+    var $this = $(this)
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    clearMenus()
+
+    if (!isActive) {
+      if ('ontouchstart' in document.documentElement && !$parent.closest('.navbar-nav').length) {
+        // if mobile we we use a backdrop because click events don't delegate
+        $('<div class="dropdown-backdrop"/>').insertAfter($(this)).on('click', clearMenus)
+      }
+
+      $parent.trigger(e = $.Event('show.bs.dropdown'))
+
+      if (e.isDefaultPrevented()) return
+
+      $parent
+        .toggleClass('open')
+        .trigger('shown.bs.dropdown')
+
+      $this.focus()
+    }
+
+    return false
+  }
+
+  Dropdown.prototype.keydown = function (e) {
+    if (!/(38|40|27)/.test(e.keyCode)) return
+
+    var $this = $(this)
+
+    e.preventDefault()
+    e.stopPropagation()
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    if (!isActive || (isActive && e.keyCode == 27)) {
+      if (e.which == 27) $parent.find(toggle).focus()
+      return $this.click()
+    }
+
+    var $items = $('[role=menu] li:not(.divider):visible a', $parent)
+
+    if (!$items.length) return
+
+    var index = $items.index($items.filter(':focus'))
+
+    if (e.keyCode == 38 && index > 0)                 index--                        // up
+    if (e.keyCode == 40 && index < $items.length - 1) index++                        // down
+    if (!~index)                                      index=0
+
+    $items.eq(index).focus()
+  }
+
+  function clearMenus() {
+    $(backdrop).remove()
+    $(toggle).each(function (e) {
+      var $parent = getParent($(this))
+      if (!$parent.hasClass('open')) return
+      $parent.trigger(e = $.Event('hide.bs.dropdown'))
+      if (e.isDefaultPrevented()) return
+      $parent.removeClass('open').trigger('hidden.bs.dropdown')
+    })
+  }
+
+  function getParent($this) {
+    var selector = $this.attr('data-target')
+
+    if (!selector) {
+      selector = $this.attr('href')
+      selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
+    }
+
+    var $parent = selector && $(selector)
+
+    return $parent && $parent.length ? $parent : $this.parent()
+  }
+
+
+  // DROPDOWN PLUGIN DEFINITION
+  // ==========================
+
+  var old = $.fn.dropdown
+
+  $.fn.dropdown = function (option) {
+    return this.each(function () {
+      var $this = $(this)
+      var data  = $this.data('dropdown')
+
+      if (!data) $this.data('dropdown', (data = new Dropdown(this)))
+      if (typeof option == 'string') data[option].call($this)
+    })
+  }
+
+  $.fn.dropdown.Constructor = Dropdown
+
+
+  // DROPDOWN NO CONFLICT
+  // ====================
+
+  $.fn.dropdown.noConflict = function () {
+    $.fn.dropdown = old
+    return this
+  }
+
+
+  // APPLY TO STANDARD DROPDOWN ELEMENTS
+  // ===================================
+
+  $(document)
+    .on('click.bs.dropdown.data-api', clearMenus)
+    .on('click.bs.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
+    .on('click.bs.dropdown.data-api'  , toggle, Dropdown.prototype.toggle)
+    .on('keydown.bs.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown)
+
+}(window.jQuery);
\ No newline at end of file
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 static/style/Gruntfile.js
--- a/static/style/Gruntfile.js
+++ b/static/style/Gruntfile.js
@@ -6,7 +6,7 @@
   var theme = grunt.option( 'theme', 'blue' );
 
   var out = 'blue'
-  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'trackster' ];
+  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'new_library', 'trackster' ];
 
   var _ = grunt.util._; 
   var fmt = _.sprintf; 
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 static/style/src/less/library.less
--- a/static/style/src/less/library.less
+++ b/static/style/src/less/library.less
@@ -29,6 +29,50 @@
     background-color: @table-bg-accent;
 }
 
+
+tr.light td
+{
+    background-color: white;
+    color: black;
+}
+tr.light:hover td
+{
+    background-color: #f5f5f5;
+    color: #8389a1;
+}
+tr.dark td
+{
+    background-color: #8389a1;
+    color: white;
+}
+tr.dark:hover td
+{
+    background-color: #bbbfd0;
+    color: white;
+}
+a.dark:hover
+{
+    color: yellow;
+    // text-decoration: none;
+}
+a.dark
+{
+    color: white;
+    // text-decoration: none;
+}
+th.button_heading
+{
+    width: 7em;
+}
+div.library_breadcrumb{
+    padding-top: 0.8em;
+    padding-bottom: 0.8em;
+}
+div.library_breadcrumb a:hover{
+    color:green;
+}
+
+
 img.expanderIcon {
     padding-right: 4px;
 }
diff -r d5217f01960360a9d7298fa679b839c806cab893 -r e2b8d3481d668203bfad7efe306925c72c644839 templates/base/base_panels.mako
--- a/templates/base/base_panels.mako
+++ b/templates/base/base_panels.mako
@@ -357,4 +357,4 @@
     ## Scripts can be loaded later since they progressively add features to
     ## the panels, but do not change layout
     ${self.late_javascripts()}
-</html>
+</html>
\ No newline at end of file
https://bitbucket.org/galaxy/galaxy-central/commits/fb3161b21a4b/
Changeset:   fb3161b21a4b
User:        martenson
Date:        2013-12-11 00:12:14
Summary:     aiding the merge of data_library branch
Affected #:  5 files
diff -r e2b8d3481d668203bfad7efe306925c72c644839 -r fb3161b21a4bfc1c5bd2f6bea8dc63c9c5bb4b38 .hgignore
--- a/.hgignore
+++ b/.hgignore
@@ -114,8 +114,4 @@
 *.orig
 .DS_Store
 *.rej
-*~
-
-
-syntax: regexp
-^static/AAA_scratch$
\ No newline at end of file
+*~
\ No newline at end of file
diff -r e2b8d3481d668203bfad7efe306925c72c644839 -r fb3161b21a4bfc1c5bd2f6bea8dc63c9c5bb4b38 lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -100,8 +100,6 @@
             encoded_id = trans.security.encode_id( content_item.id )
             time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
             time_created = content_item.create_time.strftime( "%Y-%m-%d %I:%M %p" )
-#             log.debug('XXXXXXXXXXXXXXXXXXX api type: ' + str(content_item.api_type))
-#             log.debug('XXXXXXXX ALL: ' + str(content_item.__dict__))
             
             # For folder return also hierarchy values
             if content_item.api_type == 'folder':
@@ -110,7 +108,6 @@
                 return_item.update ( dict ( item_count = content_item.item_count ) )
 
             if content_item.api_type == 'file':
-#                 log.debug('XXXXX content item class: ' + str(content_item.__class__))
                 library_dataset_dict = content_item.to_dict()
                 library_dataset_dict['data_type']
                 library_dataset_dict['file_size']
diff -r e2b8d3481d668203bfad7efe306925c72c644839 -r fb3161b21a4bfc1c5bd2f6bea8dc63c9c5bb4b38 static/scripts/galaxy.library.router.js
--- a/static/scripts/galaxy.library.router.js
+++ /dev/null
@@ -1,21 +0,0 @@
-define( [], function() {
-
-/**
- * -- Routers --
- */
-
-/**
- * Router for library browser.
- */
-var LibraryRouter = Backbone.Router.extend({    
-    routes: {
-        "" :                 "libraries",
-        "folders/:id" :      "folder_content"
-    }
-});
-
-return {
-    LibraryRouter: LibraryRouter,
-};
-
-})
\ No newline at end of file
diff -r e2b8d3481d668203bfad7efe306925c72c644839 -r fb3161b21a4bfc1c5bd2f6bea8dc63c9c5bb4b38 static/scripts/galaxy.menu.js
--- a/static/scripts/galaxy.menu.js
+++ b/static/scripts/galaxy.menu.js
@@ -57,7 +57,6 @@
 
         tab_shared.add({
             title   : "Data Libraries",
-            content : "library/index",
             content : "library/index"
         });
 
diff -r e2b8d3481d668203bfad7efe306925c72c644839 -r fb3161b21a4bfc1c5bd2f6bea8dc63c9c5bb4b38 static/style/Gruntfile.js
--- a/static/style/Gruntfile.js
+++ b/static/style/Gruntfile.js
@@ -6,7 +6,7 @@
   var theme = grunt.option( 'theme', 'blue' );
 
   var out = 'blue'
-  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'new_library', 'trackster' ];
+  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'trackster' ];
 
   var _ = grunt.util._; 
   var fmt = _.sprintf; 
https://bitbucket.org/galaxy/galaxy-central/commits/96f52c9fd562/
Changeset:   96f52c9fd562
User:        martenson
Date:        2013-12-11 00:21:06
Summary:     Merged galaxy/galaxy-central into default
Affected #:  3 files
diff -r fb3161b21a4bfc1c5bd2f6bea8dc63c9c5bb4b38 -r 96f52c9fd562fe160fc8ac71aa6cddd902d23e0f config/plugins/visualizations/scatterplot/templates/scatterplot.mako
--- a/config/plugins/visualizations/scatterplot/templates/scatterplot.mako
+++ b/config/plugins/visualizations/scatterplot/templates/scatterplot.mako
@@ -15,7 +15,6 @@
 <script type="text/javascript" src="/static/scripts/libs/jquery/jquery.migrate.js"></script><script type="text/javascript" src="/static/scripts/libs/underscore.js"></script><script type="text/javascript" src="/static/scripts/libs/backbone/backbone.js"></script>
-<script type="text/javascript" src="/static/scripts/libs/backbone/backbone-relational.js"></script><script type="text/javascript" src="/static/scripts/libs/handlebars.runtime.js"></script><script type="text/javascript" src="/static/scripts/libs/d3.js"></script><script type="text/javascript" src="/static/scripts/libs/bootstrap.js"></script>
diff -r fb3161b21a4bfc1c5bd2f6bea8dc63c9c5bb4b38 -r 96f52c9fd562fe160fc8ac71aa6cddd902d23e0f lib/galaxy/web/base/controller.py
--- a/lib/galaxy/web/base/controller.py
+++ b/lib/galaxy/web/base/controller.py
@@ -474,6 +474,9 @@
         """Returns history data in the form of a dictionary.
         """
         history_dict = history.to_dict( view='element', value_mapper={ 'id':trans.security.encode_id })
+        history_dict[ 'user_id' ] = None
+        if history.user_id:
+            history_dict[ 'user_id' ] = trans.security.encode_id( history.user_id )
 
         history_dict[ 'nice_size' ] = history.get_disk_size( nice_size=True )
         history_dict[ 'annotation' ] = history.get_item_annotation_str( trans.sa_session, trans.user, history )
diff -r fb3161b21a4bfc1c5bd2f6bea8dc63c9c5bb4b38 -r 96f52c9fd562fe160fc8ac71aa6cddd902d23e0f lib/tool_shed/galaxy_install/tool_dependencies/install_util.py
--- a/lib/tool_shed/galaxy_install/tool_dependencies/install_util.py
+++ b/lib/tool_shed/galaxy_install/tool_dependencies/install_util.py
@@ -93,7 +93,7 @@
                         env_file_builder.append_line( action="source", value=required_tool_dependency_env_file_path )
                         return_code = env_file_builder.return_code
                         if return_code:
-                            error_message = 'Error defining env.sh file for package %s, return_code: %s' % ( str( package_name, str( return_code ) ) )
+                            error_message = 'Error defining env.sh file for package %s, return_code: %s' % ( str( package_name ), str( return_code ) )
                             tool_dependency = tool_dependency_util.handle_tool_dependency_installation_error( app,
                                                                                                               tool_dependency,
                                                                                                               error_message,
https://bitbucket.org/galaxy/galaxy-central/commits/42e74befc796/
Changeset:   42e74befc796
User:        martenson
Date:        2013-12-11 00:29:50
Summary:     aid after merge data_library branch
Affected #:  1 file
diff -r 96f52c9fd562fe160fc8ac71aa6cddd902d23e0f -r 42e74befc7962afb93baba6197273565755e086c lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -73,6 +73,8 @@
     webapp.add_route( '/datasets/:dataset_id/display/{filename:.+?}', controller='dataset', action='display', dataset_id=None, filename=None)
     webapp.add_route( '/datasets/:dataset_id/:action/:filename', controller='dataset', action='index', dataset_id=None, filename=None)
     webapp.add_route( '/display_application/:dataset_id/:app_name/:link_name/:user_id/:app_action/:action_param',
+                              controller='dataset', action='display_application', dataset_id=None, user_id=None,
+                              app_name = None, link_name = None, app_action = None, action_param = None )
     webapp.add_route( '/u/:username/d/:slug/:filename', controller='dataset', action='display_by_username_and_slug', filename=None )
     webapp.add_route( '/u/:username/p/:slug', controller='page', action='display_by_username_and_slug' )
     webapp.add_route( '/u/:username/h/:slug', controller='history', action='display_by_username_and_slug' )
https://bitbucket.org/galaxy/galaxy-central/commits/8fcfddb3e1b5/
Changeset:   8fcfddb3e1b5
User:        martenson
Date:        2013-12-11 16:41:39
Summary:     aiding problematic merging of data_library branch, removing duplicates, removing API v2 relicts
Affected #:  3 files
diff -r 42e74befc7962afb93baba6197273565755e086c -r 8fcfddb3e1b57ad627ebab3619920552a6e967ff .hgignore
--- a/.hgignore
+++ b/.hgignore
@@ -114,4 +114,5 @@
 *.orig
 .DS_Store
 *.rej
-*~
\ No newline at end of file
+*~
+
diff -r 42e74befc7962afb93baba6197273565755e086c -r 8fcfddb3e1b57ad627ebab3619920552a6e967ff lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -46,7 +46,6 @@
     atexit.register( app.shutdown )
     # Create the universe WSGI application
     webapp = GalaxyWebApplication( app, session_cookie='galaxysession', name='galaxy' )
-    # Add the controllers folder
     # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
     webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
                       controller='admin_toolshed',
@@ -54,17 +53,11 @@
                       repository_id=None,
                       image_file=None )
     webapp.add_ui_controllers( 'galaxy.webapps.galaxy.controllers', app )
-    # Add the api folder
-    webapp.add_api_controllers( 'galaxy.webapps.galaxy.api', app )
-    # Add the api folder VERSION 2
-    webapp.add_api_controllers( 'galaxy.webapps.galaxy.api.v2', app )
-
     # Force /history to go to /root/history -- needed since the tests assume this
     webapp.add_route( '/history', controller='root', action='history' )
     # Force /activate to go to the controller
     webapp.add_route( '/activate', controller='user', action='activate' )
     # These two routes handle our simple needs at the moment
-    
     webapp.add_route( '/async/:tool_id/:data_id/:data_secret', controller='async', action='index', tool_id=None, data_id=None, data_secret=None )
     webapp.add_route( '/:controller/:action', action='index' )
     webapp.add_route( '/:action', controller='root', action='index' )
@@ -73,8 +66,8 @@
     webapp.add_route( '/datasets/:dataset_id/display/{filename:.+?}', controller='dataset', action='display', dataset_id=None, filename=None)
     webapp.add_route( '/datasets/:dataset_id/:action/:filename', controller='dataset', action='index', dataset_id=None, filename=None)
     webapp.add_route( '/display_application/:dataset_id/:app_name/:link_name/:user_id/:app_action/:action_param',
-                              controller='dataset', action='display_application', dataset_id=None, user_id=None,
-                              app_name = None, link_name = None, app_action = None, action_param = None )
+        controller='dataset', action='display_application', dataset_id=None, user_id=None,
+        app_name = None, link_name = None, app_action = None, action_param = None )
     webapp.add_route( '/u/:username/d/:slug/:filename', controller='dataset', action='display_by_username_and_slug', filename=None )
     webapp.add_route( '/u/:username/p/:slug', controller='page', action='display_by_username_and_slug' )
     webapp.add_route( '/u/:username/h/:slug', controller='history', action='display_by_username_and_slug' )
@@ -82,22 +75,12 @@
     webapp.add_route( '/u/:username/v/:slug', controller='visualization', action='display_by_username_and_slug' )
     webapp.add_route( '/search', controller='search', action='index' )
 
-    # Add the web API
+    # ================
+    # =====  API =====
+    # ================
+
     webapp.add_api_controllers( 'galaxy.webapps.galaxy.api', app )
-    # The /folders section is experimental at this point:
-    log.debug( "app.config.api_folders: %s" % app.config.api_folders )
-    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
-    webapp.mapper.resource( 'content', 'contents',
-                                controller='folder_contents',
-                                name_prefix='folder_',
-                                path_prefix='/api/folders/:folder_id',
-                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
-    webapp.mapper.resource( 'content',
-                                'contents',
-                                controller='library_contents',
-                                name_prefix='library_',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+
     webapp.mapper.resource( 'content',
                                 'contents',
                                 controller='history_contents',
@@ -109,10 +92,6 @@
                               controller="datasets",
                               action="display",
                               conditions=dict(method=["GET"]))
-    webapp.mapper.resource( 'permission',
-                                'permissions',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
     webapp.mapper.resource( 'user',
                                 'users',
                                 controller='group_users',
@@ -134,11 +113,6 @@
     _add_item_tags_controller( webapp,
                                name_prefix="workflow_",
                                path_prefix='/api/workflows/:workflow_id' )
-
-    _add_item_extended_metadata_controller( webapp,
-                               name_prefix="library_dataset_",
-                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
-
     _add_item_annotation_controller( webapp,
                                name_prefix="history_content_",
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
@@ -148,7 +122,6 @@
     _add_item_annotation_controller( webapp,
                                name_prefix="workflow_",
                                path_prefix='/api/workflows/:workflow_id' )
-
     _add_item_provenance_controller( webapp,
                                name_prefix="history_content_",
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
@@ -178,13 +151,12 @@
     #webapp.mapper.connect( 'run_workflow', '/api/workflow/{workflow_id}/library/{library_id}', controller='workflows', action='run', workflow_id=None, library_id=None, conditions=dict(method=["GET"]) )
     webapp.mapper.resource( 'search', 'search', path_prefix='/api' )
 
-    webapp.mapper.resource_with_deleted( 'quota', 'quotas', path_prefix='/api' )
-    webapp.mapper.resource_with_deleted( 'history', 'histories', path_prefix='/api' )
-    webapp.mapper.resource_with_deleted( 'user', 'users', path_prefix='/api' )
-
     # =======================
     # ===== LIBRARY API =====
     # =======================
+    
+    # The /folders section is experimental at this point:
+    log.debug( "app.config.api_folders: %s" % app.config.api_folders )
 
     webapp.mapper.connect( 'show_lda_item', '/api/libraries/datasets/:id', controller='lda_datasets', action='show', conditions=dict(method=["GET"]) )
     webapp.mapper.connect( 'download_lda_items', '/api/libraries/datasets/download/:format', controller='lda_datasets', action='download', conditions=dict(method=["POST"]) )
@@ -232,6 +204,9 @@
     # Preserve the following download route for now for dependent applications  -- deprecate at some point
     webapp.mapper.connect("workflow_dict", '/api/workflows/download/{workflow_id}', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
     
+    # ====================
+    # ===== TOOLSHED =====
+    # ====================
     
     # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
     webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
@@ -239,7 +214,7 @@
                       action='display_image_in_repository',
                       repository_id=None,
                       image_file=None )
-
+    
     # Galaxy API for tool shed features.
     webapp.mapper.resource( 'tool_shed_repository',
                             'tool_shed_repositories',
@@ -253,6 +228,7 @@
                             path_prefix='/api',
                             new={ 'install_repository_revision' : 'POST' },
                             parent_resources=dict( member_name='tool_shed_repository', collection_name='tool_shed_repositories' ) )
+
     # Connect logger from app
     if app.trace_logger:
         webapp.trace_logger = app.trace_logger
@@ -273,11 +249,6 @@
         galaxy.model.mapping.metadata.engine.connection_provider._pool.dispose()
     except:
         pass
-    # Close any pooled database connections before forking
-    try:
-        galaxy.model.tool_shed_install.mapping.metadata.engine.connection_provider._pool.dispose()
-    except:
-        pass
 
     # Return
     return webapp
https://bitbucket.org/galaxy/galaxy-central/commits/4506583cb1bc/
Changeset:   4506583cb1bc
User:        martenson
Date:        2013-12-11 16:51:21
Summary:     aiding merg problems with data_library branch
Affected #:  1 file
diff -r 8fcfddb3e1b57ad627ebab3619920552a6e967ff -r 4506583cb1bc66db01d98c1619b6829fcda31e00 lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -46,12 +46,6 @@
     atexit.register( app.shutdown )
     # Create the universe WSGI application
     webapp = GalaxyWebApplication( app, session_cookie='galaxysession', name='galaxy' )
-    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
-    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
-                      controller='admin_toolshed',
-                      action='display_image_in_repository',
-                      repository_id=None,
-                      image_file=None )
     webapp.add_ui_controllers( 'galaxy.webapps.galaxy.controllers', app )
     # Force /history to go to /root/history -- needed since the tests assume this
     webapp.add_route( '/history', controller='root', action='history' )
@@ -187,7 +181,6 @@
     # add as a non-ATOM API call to support the notion of a 'current/working' history unique to the history resource
     webapp.mapper.connect( "set_as_current", "/api/histories/{id}/set_as_current",
         controller="histories", action="set_as_current", conditions=dict( method=["POST"] ) )
-    webapp.mapper.connect( "set_as_current", "/api/histories/{id}/set_as_current", controller="histories", action="set_as_current", conditions=dict( method=["POST"] ) )
 
     webapp.mapper.connect( "create_api_key", "/api/users/:user_id/api_key",
         controller="users", action="api_key", user_id=None, conditions=dict( method=["POST"] ) )
@@ -195,7 +188,6 @@
     # visualizations registry generic template renderer
     webapp.add_route( '/visualization/show/:visualization_name',
         controller='visualization', action='render', visualization_name=None )
-    webapp.add_route( '/visualization/show/:visualization_name', controller='visualization', action='render', visualization_name=None )
 
     # "POST /api/workflows/import"  =>  ``workflows.import_workflow()``.
     # Defines a named route "import_workflow".
https://bitbucket.org/galaxy/galaxy-central/commits/2fb8dfdb5b19/
Changeset:   2fb8dfdb5b19
User:        martenson
Date:        2013-12-11 16:53:42
Summary:     aiding merg problems with data_library branch
Affected #:  1 file
diff -r 4506583cb1bc66db01d98c1619b6829fcda31e00 -r 2fb8dfdb5b19378a5bc61ee713aa4da3f9767a91 lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -241,6 +241,11 @@
         galaxy.model.mapping.metadata.engine.connection_provider._pool.dispose()
     except:
         pass
+     # Close any pooled database connections before forking
+    try:
+        galaxy.model.tool_shed_install.mapping.metadata.engine.connection_provider._pool.dispose()
+    except:
+        pass
 
     # Return
     return webapp
https://bitbucket.org/galaxy/galaxy-central/commits/78c39d32ac09/
Changeset:   78c39d32ac09
User:        martenson
Date:        2013-12-11 17:01:45
Summary:     aiding merg problems with data_library branch
Affected #:  1 file
diff -r 2fb8dfdb5b19378a5bc61ee713aa4da3f9767a91 -r 78c39d32ac09a796966cc192636f4f8b1ba1bc07 templates/base/base_panels.mako
--- a/templates/base/base_panels.mako
+++ b/templates/base/base_panels.mako
@@ -357,4 +357,4 @@
     ## Scripts can be loaded later since they progressively add features to
     ## the panels, but do not change layout
     ${self.late_javascripts()}
-</html>
\ No newline at end of file
+</html>
https://bitbucket.org/galaxy/galaxy-central/commits/8528640574a5/
Changeset:   8528640574a5
User:        martenson
Date:        2013-12-11 17:08:59
Summary:     aiding merg problems with data_library branch
Affected #:  1 file
diff -r 78c39d32ac09a796966cc192636f4f8b1ba1bc07 -r 8528640574a5b981ffb0940477abb0bd65ea2885 static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -217,11 +217,12 @@
         return  '<div class="modal">' +
                     '<div class="modal-backdrop fade in" style="z-index: -1;"></div>' +
                     '<div class="modal-dialog">' +
+                        '<div class="modal-content">' +
                             '<div class="modal-header">' +
                                 '<button type="button" class="close" style="display: none;">×</button>' +
                                 '<h4 class="title">' + title + '</h4>' +
                                 '<span class="notification-modal"></span>' + 
-                            '</div>' +
+                                '</div>' +
                             '<div class="modal-body"></div>' +
                             '<div class="modal-footer">' +
                                 '<div class="buttons" style="float: right;"></div>' +
https://bitbucket.org/galaxy/galaxy-central/commits/181b293ea659/
Changeset:   181b293ea659
User:        martenson
Date:        2013-12-11 17:19:42
Summary:     Merge with second head of ‘default’ branch
Affected #:  17 files
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 lib/galaxy/model/__init__.py
--- a/lib/galaxy/model/__init__.py
+++ b/lib/galaxy/model/__init__.py
@@ -1827,7 +1827,7 @@
 class Library( object, Dictifiable ):
     permitted_actions = get_permitted_actions( filter='LIBRARY' )
     dict_collection_visible_keys = ( 'id', 'name' )
-    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis' )
+    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis', 'root_folder_id' )
     def __init__( self, name=None, description=None, synopsis=None, root_folder=None ):
         self.name = name or "Unnamed library"
         self.description = description
@@ -1894,7 +1894,7 @@
         return name
 
 class LibraryFolder( object, Dictifiable ):
-    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build' )
+    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build', 'update_time' )
     def __init__( self, name=None, description=None, item_count=0, order_id=None ):
         self.name = name or "Unnamed folder"
         self.description = description
@@ -2060,6 +2060,7 @@
                      genome_build = ldda.dbkey,
                      misc_info = ldda.info,
                      misc_blurb = ldda.blurb,
+                     peek = ( lambda ldda: ldda.display_peek() if ldda.peek and ldda.peek != 'no peek' else None )( ldda ),
                      template_data = template_data )
         if ldda.dataset.uuid is None:
             rval['uuid'] = None
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 lib/galaxy/model/mapping.py
--- a/lib/galaxy/model/mapping.py
+++ b/lib/galaxy/model/mapping.py
@@ -1878,8 +1878,9 @@
     table = self.table
     trans = conn.begin()
     try:
-        next_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
-        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid + 1 ) )
+        current_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
+        next_hid = current_hid + 1
+        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid ) )
         trans.commit()
         return next_hid
     except:
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 lib/galaxy/webapps/galaxy/api/datasets.py
--- a/lib/galaxy/webapps/galaxy/api/datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/datasets.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a dataset.
+API operations on the contents of a history dataset.
 """
 from galaxy import web
 from galaxy.visualization.data_providers.genome import FeatureLocationIndexDataProvider
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a library.
+API operations on the contents of a folder.
 """
 import logging, os, string, shutil, urllib, re, socket
 from cgi import escape, FieldStorage
@@ -11,67 +11,122 @@
 log = logging.getLogger( __name__ )
 
 class FolderContentsController( BaseAPIController, UsesLibraryMixin, UsesLibraryMixinItems ):
+    """
+    Class controls retrieval, creation and updating of folder contents.
+    """
+
+    def load_folder_contents( self, trans, folder ):
+        """
+        Loads all contents of the folder (folders and data sets) but only in the first level.
+        """
+        current_user_roles = trans.get_current_user_roles()
+        is_admin = trans.user_is_admin()
+        content_items = []
+        for subfolder in folder.active_folders:
+            if not is_admin:
+                can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
+            if (is_admin or can_access) and not subfolder.deleted:
+                subfolder.api_type = 'folder'
+                content_items.append( subfolder )
+        for dataset in folder.datasets:
+            if not is_admin:
+                can_access = trans.app.security_agent.can_access_dataset( current_user_roles, dataset.library_dataset_dataset_association.dataset )
+            if (is_admin or can_access) and not dataset.deleted:
+                dataset.api_type = 'file'
+                content_items.append( dataset )
+        return content_items
 
     @web.expose_api
     def index( self, trans, folder_id, **kwd ):
         """
         GET /api/folders/{encoded_folder_id}/contents
         Displays a collection (list) of a folder's contents (files and folders).
-        The /api/library_contents/{encoded_library_id}/contents
-        lists everything in a library recursively, which is not what
-        we want here. We could add a parameter to use the recursive
-        style, but this is meant to act similar to an "ls" directory listing.
+        Encoded folder ID is prepended with 'F' if it is a folder as opposed to a data set which does not have it.
+        Full path is provided as a separate object in response providing data for breadcrumb path building.
         """
-        rval = []
+        folder_container = []
         current_user_roles = trans.get_current_user_roles()
 
-        def traverse( folder ):
-            admin = trans.user_is_admin()
-            rval = []
-            for subfolder in folder.active_folders:
-                if not admin:
-                    can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
-                if (admin or can_access) and not subfolder.deleted:
-                    subfolder.api_type = 'folder'
-                    rval.append( subfolder )
-            for ld in folder.datasets:
-                if not admin:
-                    can_access = trans.app.security_agent.can_access_dataset( current_user_roles, ld.library_dataset_dataset_association.dataset )
-                if (admin or can_access) and not ld.deleted:
-                    ld.api_type = 'file'
-                    rval.append( ld )
-            return rval
-
-        try:
-            decoded_folder_id = trans.security.decode_id( folder_id[-16:] )
-        except TypeError:
-            trans.response.status = 400
-            return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
+        if ( folder_id.startswith( 'F' ) ):
+            try:
+                decoded_folder_id = trans.security.decode_id( folder_id[1:] )
+            except TypeError:
+                trans.response.status = 400
+                return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
 
         try:
             folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( decoded_folder_id )
-            parent_library = folder.parent_library
         except:
             folder = None
-            log.error( "FolderContentsController.index: Unable to retrieve folder %s"
-                      % folder_id )
+            log.error( "FolderContentsController.index: Unable to retrieve folder with ID: %s" % folder_id )
 
-        # TODO: Find the API's path to this folder if necessary.
-        # This was needed in recursive descent, but it's not needed
-        # for "ls"-style content checking:
-        if not folder or not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+        # We didn't find the folder or user does not have an access to it.
+        if not folder:
             trans.response.status = 400
             return "Invalid folder id ( %s ) specified." % str( folder_id )
+        
+        if not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+            log.warning( "SECURITY: User (id: %s) without proper access rights is trying to load folder with ID of %s" % ( trans.user.id, folder.id ) )
+            trans.response.status = 400
+            return "Invalid folder id ( %s ) specified." % str( folder_id )
+        
+        path_to_root = []
+        def build_path ( folder ):
+            """
+            Search the path upwards recursively and load the whole route of names and ids for breadcrumb purposes.
+            """
+            path_to_root = []
+            # We are almost in root
+            if folder.parent_id is None:
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
+            else:
+            # We add the current folder and traverse up one folder.
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
+                upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_id )
+                path_to_root.extend( build_path( upper_folder ) )
+            return path_to_root
+            
+        # Return the reversed path so it starts with the library node.
+        full_path = build_path( folder )[::-1]
+        folder_container.append( dict( full_path = full_path ) )
+        
+        folder_contents = []
+        time_updated = ''
+        time_created = ''
+        # Go through every item in the folder and include its meta-data.
+        for content_item in self.load_folder_contents( trans, folder ):
+#             rval = content_item.to_dict()
+            return_item = {}
+            encoded_id = trans.security.encode_id( content_item.id )
+            time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+            time_created = content_item.create_time.strftime( "%Y-%m-%d %I:%M %p" )
+            
+            # For folder return also hierarchy values
+            if content_item.api_type == 'folder':
+                encoded_id = 'F' + encoded_id
+#                 time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+                return_item.update ( dict ( item_count = content_item.item_count ) )
 
-        for content in traverse( folder ):
-            encoded_id = trans.security.encode_id( content.id )
-            if content.api_type == 'folder':
-                encoded_id = 'F' + encoded_id
-            rval.append( dict( id = encoded_id,
-                               type = content.api_type,
-                               name = content.name,
-                               url = url_for( 'folder_contents', folder_id=encoded_id ) ) )
-        return rval
+            if content_item.api_type == 'file':
+                library_dataset_dict = content_item.to_dict()
+                library_dataset_dict['data_type']
+                library_dataset_dict['file_size']
+                library_dataset_dict['date_uploaded']
+                return_item.update ( dict ( data_type = library_dataset_dict['data_type'],
+                                            file_size = library_dataset_dict['file_size'],
+                                            date_uploaded = library_dataset_dict['date_uploaded'] ) )
+
+            # For every item return also the default meta-data
+            return_item.update( dict( id = encoded_id,
+                               type = content_item.api_type,
+                               name = content_item.name,
+                               time_updated = time_updated,
+                               time_created = time_created
+                                ) )
+            folder_contents.append( return_item )
+        # Put the data in the container
+        folder_container.append( dict( folder_contents = folder_contents ) )
+        return folder_container
 
     @web.expose_api
     def show( self, trans, id, library_id, **kwd ):
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- /dev/null
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -0,0 +1,241 @@
+"""
+API operations on the dataset from library.
+"""
+import glob
+import logging
+import operator
+import os
+import os.path
+import string
+import sys
+import tarfile
+import tempfile
+import urllib
+import urllib2
+import zipfile
+from galaxy.security import Action
+from galaxy import util, web
+from galaxy.util.streamball import StreamBall
+from galaxy.web.base.controller import BaseAPIController, UsesLibraryMixinItems
+
+import logging
+log = logging.getLogger( __name__ )
+
+# Test for available compression types
+# tmpd = tempfile.mkdtemp()
+# comptypes = []
+# for comptype in ( 'gz', 'bz2' ):
+#     tmpf = os.path.join( tmpd, 'compression_test.tar.' + comptype )
+#     try:
+#         archive = tarfile.open( tmpf, 'w:' + comptype )
+#         archive.close()
+#         comptypes.append( comptype )
+#     except tarfile.CompressionError:
+#         log.exception( "Compression error when testing %s compression.  This option will be disabled for library downloads." % comptype )
+#     try:
+#         os.unlink( tmpf )
+#     except OSError:
+#         pass
+ziptype = '32'
+# tmpf = os.path.join( tmpd, 'compression_test.zip' )
+# try:
+#     archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
+#     archive.close()
+#     comptypes.append( 'zip' )
+#     ziptype = '64'
+# except RuntimeError:
+#     log.exception( "Compression error when testing zip compression. This option will be disabled for library downloads." )
+# except (TypeError, zipfile.LargeZipFile):
+#     # ZIP64 is only in Python2.5+.  Remove TypeError when 2.4 support is dropped
+#     log.warning( 'Max zip file size is 2GB, ZIP64 not supported' )
+#     comptypes.append( 'zip' )
+# try:
+#     os.unlink( tmpf )
+# except OSError:
+#     pass
+# os.rmdir( tmpd )
+
+
+
+class DatasetsController( BaseAPIController, UsesLibraryMixinItems ):
+
+    @web.expose_api
+    def show( self, trans, id, **kwd ):
+        """
+        GET /api/libraries/datasets/{encoded_dataset_id}
+        Displays information about the dataset identified by the lda ID.
+        """
+        # Get dataset.
+        try:
+            dataset = self.get_library_dataset( trans, id = id )
+        except Exception, e:
+            return str( e )
+        try:
+            # Default: return dataset as dict.
+            rval = dataset.to_dict()
+        except Exception, e:
+            rval = "Error in dataset API at listing contents: " + str( e )
+            log.error( rval + ": %s" % str(e), exc_info=True )
+            trans.response.status = 500
+        
+        rval['id'] = trans.security.encode_id(rval['id']);
+        rval['ldda_id'] = trans.security.encode_id(rval['ldda_id']);
+        rval['folder_id'] = 'f' + trans.security.encode_id(rval['folder_id'])
+        
+        return rval
+    
+    @web.expose
+    def download( self, trans, format, **kwd ):
+        """
+        POST /api/libraries/datasets/download/{format}
+        POST data: ldda_ids = []
+        Downloads dataset(s) in the requested format.
+        """
+        lddas = []
+#         is_admin = trans.user_is_admin()
+#         current_user_roles = trans.get_current_user_roles()
+        
+        datasets_to_download = kwd['ldda_ids%5B%5D']
+        
+        if ( datasets_to_download != None ):
+            datasets_to_download = util.listify( datasets_to_download )
+            for dataset_id in datasets_to_download:
+                try:
+                    ldda = trans.sa_session.query( trans.app.model.LibraryDatasetDatasetAssociation ).get( trans.security.decode_id( dataset_id ) )
+                    assert not ldda.dataset.purged
+                    lddas.append( ldda )
+                except:
+                    ldda = None
+                    message += "Invalid library dataset id (%s) specified.  " % str( dataset_id )
+                    
+        if format in [ 'zip','tgz','tbz' ]:
+                error = False
+                killme = string.punctuation + string.whitespace
+                trantab = string.maketrans(killme,'_'*len(killme))
+                try:
+                    outext = 'zip'
+                    if format == 'zip':
+                        # Can't use mkstemp - the file must not exist first
+                        tmpd = tempfile.mkdtemp()
+                        util.umask_fix_perms( tmpd, trans.app.config.umask, 0777, self.app.config.gid )
+                        tmpf = os.path.join( tmpd, 'library_download.' + format )
+                        if ziptype == '64' and trans.app.config.upstream_gzip:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED, True )
+                        elif ziptype == '64':
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
+                        elif trans.app.config.upstream_gzip:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED )
+                        else:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED )
+                        archive.add = lambda x, y: archive.write( x, y.encode('CP437') )
+                    elif format == 'tgz':
+                        if trans.app.config.upstream_gzip:
+                            archive = StreamBall( 'w|' )
+                            outext = 'tar'
+                        else:
+                            archive = StreamBall( 'w|gz' )
+                            outext = 'tgz'
+                    elif format == 'tbz':
+                        archive = StreamBall( 'w|bz2' )
+                        outext = 'tbz2'
+                except ( OSError, zipfile.BadZipfile ):
+                    error = True
+                    log.exception( "Unable to create archive for download" )
+                    message = "Unable to create archive for download, please report this error"
+                    status = 'error'
+                except:
+                     error = True
+                     log.exception( "Unexpected error %s in create archive for download" % sys.exc_info()[0] )
+                     message = "Unable to create archive for download, please report - %s" % sys.exc_info()[0]
+                     status = 'error'
+                if not error:
+                    composite_extensions = trans.app.datatypes_registry.get_composite_extensions()
+                    seen = []
+                    for ldda in lddas:
+                        if ldda.dataset.state in [ 'new', 'upload', 'queued', 'running', 'empty', 'discarded' ]:
+                            continue
+                        ext = ldda.extension
+                        is_composite = ext in composite_extensions
+                        path = ""
+                        parent_folder = ldda.library_dataset.folder
+                        while parent_folder is not None:
+                            # Exclude the now-hidden "root folder"
+                            if parent_folder.parent is None:
+                                path = os.path.join( parent_folder.library_root[0].name, path )
+                                break
+                            path = os.path.join( parent_folder.name, path )
+                            parent_folder = parent_folder.parent
+                        path += ldda.name
+                        while path in seen:
+                            path += '_'
+                        seen.append( path )
+                        zpath = os.path.split(path)[-1] # comes as base_name/fname
+                        outfname,zpathext = os.path.splitext(zpath)
+                        if is_composite:
+                            # need to add all the components from the extra_files_path to the zip
+                            if zpathext == '':
+                                zpath = '%s.html' % zpath # fake the real nature of the html file
+                            try:
+                                archive.add(ldda.dataset.file_name,zpath) # add the primary of a composite set
+                            except IOError:
+                                error = True
+                                log.exception( "Unable to add composite parent %s to temporary library download archive" % ldda.dataset.file_name)
+                                message = "Unable to create archive for download, please report this error"
+                                status = 'error'
+                                continue
+                            flist = glob.glob(os.path.join(ldda.dataset.extra_files_path,'*.*')) # glob returns full paths
+                            for fpath in flist:
+                                efp,fname = os.path.split(fpath)
+                                if fname > '':
+                                    fname = fname.translate(trantab)
+                                try:
+                                    archive.add( fpath,fname )
+                                except IOError:
+                                    error = True
+                                    log.exception( "Unable to add %s to temporary library download archive %s" % (fname,outfname))
+                                    message = "Unable to create archive for download, please report this error"
+                                    status = 'error'
+                                    continue
+                        else: # simple case
+                            try:
+                                archive.add( ldda.dataset.file_name, path )
+                            except IOError:
+                                error = True
+                                log.exception( "Unable to write %s to temporary library download archive" % ldda.dataset.file_name)
+                                message = "Unable to create archive for download, please report this error"
+                                status = 'error'
+                    if not error:
+                        lname = 'selected_dataset'
+                        fname = lname.replace( ' ', '_' ) + '_files'
+                        if format == 'zip':
+                            archive.close()
+                            trans.response.set_content_type( "application/octet-stream" )
+                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                            archive = util.streamball.ZipBall(tmpf, tmpd)
+                            archive.wsgi_status = trans.response.wsgi_status()
+                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            return archive.stream
+                        else:
+                            trans.response.set_content_type( "application/x-tar" )
+                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                            archive.wsgi_status = trans.response.wsgi_status()
+                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            return archive.stream
+        elif format == 'uncompressed':
+            if len(lddas) != 1:
+                return 'Wrong request'
+            else:
+                single_dataset = lddas[0]
+                trans.response.set_content_type( single_dataset.get_mime() )
+                fStat = os.stat( ldda.file_name )
+                trans.response.headers[ 'Content-Length' ] = int( fStat.st_size )
+                valid_chars = '.,^_-()[]0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
+                fname = ldda.name
+                fname = ''.join( c in valid_chars and c or '_' for c in fname )[ 0:150 ]
+                trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s"' % fname
+                try:
+                    return open( single_dataset.file_name )
+                except:
+                    return 'This dataset contains no content'
+        else:
+            return 'Wrong format';
\ No newline at end of file
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 lib/galaxy/webapps/galaxy/api/libraries.py
--- a/lib/galaxy/webapps/galaxy/api/libraries.py
+++ b/lib/galaxy/webapps/galaxy/api/libraries.py
@@ -49,9 +49,10 @@
                            trans.model.Library.table.c.id.in_( accessible_restricted_library_ids ) ) )
         rval = []
         for library in query:
-            item = library.to_dict()
+            item = library.to_dict( view='element' )
             item['url'] = url_for( route, id=trans.security.encode_id( library.id ) )
-            item['id'] = trans.security.encode_id( item['id'] )
+            item['id'] = 'F' + trans.security.encode_id( item['id'] )
+            item['root_folder_id'] = 'F' + trans.security.encode_id( item['root_folder_id'] )
             rval.append( item )
         return rval
 
@@ -131,6 +132,9 @@
         rval['name'] = name
         rval['id'] = encoded_id
         return rval
+    
+    def edit( self, trans, payload, **kwd  ):
+        return "Not implemented yet"
 
     @web.expose_api
     def delete( self, trans, id, **kwd ):
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -46,12 +46,6 @@
     atexit.register( app.shutdown )
     # Create the universe WSGI application
     webapp = GalaxyWebApplication( app, session_cookie='galaxysession', name='galaxy' )
-    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
-    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
-                      controller='admin_toolshed',
-                      action='display_image_in_repository',
-                      repository_id=None,
-                      image_file=None )
     webapp.add_ui_controllers( 'galaxy.webapps.galaxy.controllers', app )
     # Force /history to go to /root/history -- needed since the tests assume this
     webapp.add_route( '/history', controller='root', action='history' )
@@ -75,22 +69,12 @@
     webapp.add_route( '/u/:username/v/:slug', controller='visualization', action='display_by_username_and_slug' )
     webapp.add_route( '/search', controller='search', action='index' )
 
-    # Add the web API
+    # ================
+    # =====  API =====
+    # ================
+
     webapp.add_api_controllers( 'galaxy.webapps.galaxy.api', app )
-    # The /folders section is experimental at this point:
-    log.debug( "app.config.api_folders: %s" % app.config.api_folders )
-    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
-    webapp.mapper.resource( 'content', 'contents',
-                                controller='folder_contents',
-                                name_prefix='folder_',
-                                path_prefix='/api/folders/:folder_id',
-                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
-    webapp.mapper.resource( 'content',
-                                'contents',
-                                controller='library_contents',
-                                name_prefix='library_',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+
     webapp.mapper.resource( 'content',
                                 'contents',
                                 controller='history_contents',
@@ -102,10 +86,6 @@
                               controller="datasets",
                               action="display",
                               conditions=dict(method=["GET"]))
-    webapp.mapper.resource( 'permission',
-                                'permissions',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
     webapp.mapper.resource( 'user',
                                 'users',
                                 controller='group_users',
@@ -127,11 +107,6 @@
     _add_item_tags_controller( webapp,
                                name_prefix="workflow_",
                                path_prefix='/api/workflows/:workflow_id' )
-
-    _add_item_extended_metadata_controller( webapp,
-                               name_prefix="library_dataset_",
-                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
-
     _add_item_annotation_controller( webapp,
                                name_prefix="history_content_",
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
@@ -141,7 +116,6 @@
     _add_item_annotation_controller( webapp,
                                name_prefix="workflow_",
                                path_prefix='/api/workflows/:workflow_id' )
-
     _add_item_provenance_controller( webapp,
                                name_prefix="history_content_",
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
@@ -171,6 +145,39 @@
     #webapp.mapper.connect( 'run_workflow', '/api/workflow/{workflow_id}/library/{library_id}', controller='workflows', action='run', workflow_id=None, library_id=None, conditions=dict(method=["GET"]) )
     webapp.mapper.resource( 'search', 'search', path_prefix='/api' )
 
+    # =======================
+    # ===== LIBRARY API =====
+    # =======================
+    
+    # The /folders section is experimental at this point:
+    log.debug( "app.config.api_folders: %s" % app.config.api_folders )
+
+    webapp.mapper.connect( 'show_lda_item', '/api/libraries/datasets/:id', controller='lda_datasets', action='show', conditions=dict(method=["GET"]) )
+    webapp.mapper.connect( 'download_lda_items', '/api/libraries/datasets/download/:format', controller='lda_datasets', action='download', conditions=dict(method=["POST"]) )
+
+    webapp.mapper.resource_with_deleted( 'library', 'libraries', path_prefix='/api' )
+    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
+
+    webapp.mapper.resource( 'content', 'contents',
+                                controller='folder_contents',
+                                name_prefix='folder_',
+                                path_prefix='/api/folders/:folder_id',
+                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
+    webapp.mapper.resource( 'content',
+                                'contents',
+                                controller='library_contents',
+                                name_prefix='library_',
+                                path_prefix='/api/libraries/:library_id',
+                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    webapp.mapper.resource( 'permission',
+                                'permissions',
+                                path_prefix='/api/libraries/:library_id',
+                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    _add_item_extended_metadata_controller( webapp,
+                               name_prefix="library_dataset_",
+                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
+    
+
     # add as a non-ATOM API call to support the notion of a 'current/working' history unique to the history resource
     webapp.mapper.connect( "set_as_current", "/api/histories/{id}/set_as_current",
         controller="histories", action="set_as_current", conditions=dict( method=["POST"] ) )
@@ -188,6 +195,18 @@
     webapp.mapper.connect("workflow_dict", '/api/workflows/{workflow_id}/download', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
     # Preserve the following download route for now for dependent applications  -- deprecate at some point
     webapp.mapper.connect("workflow_dict", '/api/workflows/download/{workflow_id}', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
+    
+    # ====================
+    # ===== TOOLSHED =====
+    # ====================
+    
+    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
+    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
+                      controller='admin_toolshed',
+                      action='display_image_in_repository',
+                      repository_id=None,
+                      image_file=None )
+    
     # Galaxy API for tool shed features.
     webapp.mapper.resource( 'tool_shed_repository',
                             'tool_shed_repositories',
@@ -201,6 +220,7 @@
                             path_prefix='/api',
                             new={ 'install_repository_revision' : 'POST' },
                             parent_resources=dict( member_name='tool_shed_repository', collection_name='tool_shed_repositories' ) )
+
     # Connect logger from app
     if app.trace_logger:
         webapp.trace_logger = app.trace_logger
@@ -221,7 +241,7 @@
         galaxy.model.mapping.metadata.engine.connection_provider._pool.dispose()
     except:
         pass
-    # Close any pooled database connections before forking
+     # Close any pooled database connections before forking
     try:
         galaxy.model.tool_shed_install.mapping.metadata.engine.connection_provider._pool.dispose()
     except:
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 lib/galaxy/webapps/galaxy/controllers/library.py
--- a/lib/galaxy/webapps/galaxy/controllers/library.py
+++ b/lib/galaxy/webapps/galaxy/controllers/library.py
@@ -76,6 +76,17 @@
 
     library_list_grid = LibraryListGrid()
 
+    
+    @web.expose
+    def list( self, trans, **kwd ):
+        params = util.Params( kwd )
+        # define app configuration for generic mako template
+        app = {
+            'jscript'       : "galaxy.library"
+        }
+        # fill template
+        return trans.fill_template('galaxy.panels.mako', config = {'app' : app})
+
     @web.expose
     def index( self, trans, **kwd ):
         params = util.Params( kwd )
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 static/scripts/galaxy.library.js
--- /dev/null
+++ b/static/scripts/galaxy.library.js
@@ -0,0 +1,860 @@
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+// === GALAXY LIBRARY MODULE ====
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+
+// global variables
+var view            = null;
+var library_router  = null;
+var responses = [];
+
+
+// load required libraries
+// require([
+//     // load js libraries
+//     'utils/galaxy.css',
+//     ], function(css){
+//         // load css
+//         css.load_file("static/style/library.css");
+// });
+
+// dependencies
+define(["galaxy.modal", "galaxy.masthead", "utils/galaxy.utils"], function(mod_modal, mod_masthead, mod_utils) {
+
+// MMMMMMMMMMMMMMM
+// === Models ====
+// MMMMMMMMMMMMMMM
+
+    // LIBRARY
+    var Library = Backbone.Model.extend({
+      urlRoot: '/api/libraries'
+  });
+
+    // LIBRARIES
+    var Libraries = Backbone.Collection.extend({
+      url: '/api/libraries',
+      model: Library
+  });
+
+    // ITEM
+    var Item = Backbone.Model.extend({
+        urlRoot : '/api/libraries/datasets'
+    })
+
+    // FOLDER
+    var Folder = Backbone.Collection.extend({
+      model: Item      
+  })
+
+    // CONTAINER for folder contents (folders, items and metadata).
+    var FolderContainer = Backbone.Model.extend({
+     defaults : {
+        folder : new Folder(),
+        full_path : "unknown",
+        urlRoot : "/api/folders/",
+        id : "unknown"
+    },
+    parse : function(obj) {
+      this.full_path = obj[0].full_path;
+          // update the inner collection
+          this.get("folder").reset(obj[1].folder_contents);
+          return obj;
+      }
+  })
+
+    // HISTORY ITEM
+    var HistoryItem = Backbone.Model.extend({
+        urlRoot : '/api/histories/'
+    });
+
+    // HISTORY
+    var GalaxyHistory = Backbone.Model.extend({
+        url : '/api/histories/'
+    });
+
+    // HISTORIES
+    var GalaxyHistories = Backbone.Collection.extend({
+        url : '/api/histories',
+        model : GalaxyHistory
+    });
+
+    //ROUTER
+    var LibraryRouter = Backbone.Router.extend({    
+        routes: {
+            "" :                 "libraries",
+            "folders/:id" :      "folder_content",
+            "folders/:folder_id/download/:format" : "download"
+        }
+    });
+
+
+// MMMMMMMMMMMMMM
+// === VIEWS ====
+// MMMMMMMMMMMMMM
+
+// galaxy folder
+var FolderContentView = Backbone.View.extend({
+    // main element definition
+    el : '#center',
+    // progress percentage
+    progress: 0,
+    // progress rate per one item
+    progressStep: 1, 
+    // last selected history in modal for UX
+    lastSelectedHistory: '',
+    // self modal
+    modal : null,
+    // loaded folders
+    folders : null,
+    
+    // initialize
+    initialize : function(){
+        this.folders = [];
+        this.queue = jQuery.Deferred();
+        this.queue.resolve();
+    },    
+
+// MMMMMMMMMMMMMMMMMM
+// === TEMPLATES ====
+// MMMMMMMMMMMMMMMMMM
+
+    // set up 
+    templateFolder : function (){
+        var tmpl_array = [];
+
+        // CONTAINER
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; ">');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
+
+        // TOOLBAR
+        tmpl_array.push('<div id="library_folder_toolbar" >');
+        tmpl_array.push('   <button title="Create New Folder" id="toolbtn_create_folder" class="btn btn-primary" type="button"><span class="fa fa-icon-plus"></span><span class="fa fa-icon-folder-close"></span> folder</button>');
+        tmpl_array.push('   <button id="toolbtn_bulk_import" class="btn btn-primary" style="display: none; margin-left: 0.5em;" type="button"><span class="fa fa-icon-external-link"></span> to history</button>');
+        
+        tmpl_array.push('   <div id="toolbtn_dl" class="btn-group" style="margin-left: 0.5em; display: none; ">');
+        tmpl_array.push('       <button id="drop_toggle" type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown">');
+        tmpl_array.push('       <span class="fa fa-icon-download"></span> download <span class="caret"></span>');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       <ul class="dropdown-menu" role="menu">');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tgz">.tar.gz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tbz">.tar.bz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/zip">.zip</a></li>');
+        tmpl_array.push('       </ul>');
+        tmpl_array.push('   </div>');
+
+        tmpl_array.push('</div>');
+
+        // BREADCRUMBS
+        tmpl_array.push('<div class="library_breadcrumb">');
+        tmpl_array.push('<a title="Return to the list of libraries" href="#">Libraries</a><b>|</b> ');
+        tmpl_array.push('<% _.each(path, function(path_item) { %>');
+        tmpl_array.push('<% if (path_item[0] != id) { %>');
+        tmpl_array.push('<a title="Return to this folder" href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a><b>|</b> ');
+        tmpl_array.push('<% } else { %>');
+        tmpl_array.push('<span title="You are in this folder"><%- path_item[1] %></span>');
+        tmpl_array.push('<% } %>');
+        tmpl_array.push('<% }); %>');
+        tmpl_array.push('</div>');
+
+        // FOLDER CONTENT
+        tmpl_array.push('<table id="folder_table" class="table table-condensed">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
+        tmpl_array.push('       <th class="button_heading">view</th>');
+        tmpl_array.push('       <th>name</th>');
+        tmpl_array.push('       <th>data type</th>');
+        tmpl_array.push('       <th>size</th>');
+        tmpl_array.push('       <th>date</th>');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td><button title="Go to parent folder" type="button" data-id="<%- upper_folder_id %>" class="btn_open_folder btn btn-default btn-xs">');        
+        tmpl_array.push('       <span class="fa fa-icon-arrow-up"></span> .. go up</td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       </tr>');        
+        tmpl_array.push('       <% _.each(items, function(content_item) { %>');
+        tmpl_array.push('       <tr class="folder_row light" id="<%- content_item.id %>">');
+        tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>'); // folder
+        tmpl_array.push('               <td></td>');
+        tmpl_array.push('               <td><button title="Open this folder" type="button" data-id="<%- content_item.id %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- content_item.get("name") %>');
+        tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>'); // empty folder
+        tmpl_array.push('           <span class="muted">(empty folder)</span>');
+        tmpl_array.push('           <% } %>');
+        tmpl_array.push('           </td>');
+        tmpl_array.push('           <td>folder</td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("item_count")) %> item(s)</td>'); // size
+        tmpl_array.push('           <% } else {  %>');
+        tmpl_array.push('           <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
+        tmpl_array.push('       <td>');
+        tmpl_array.push('       <button title="See details of this dataset" type="button" class="library-dataset btn btn-default btn-xs">');
+        tmpl_array.push('       <span class="fa fa-icon-eye-open"></span> details');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       </td>');
+        tmpl_array.push('           <td><%- content_item.get("name") %></td>'); // dataset
+        tmpl_array.push('           <td><%= _.escape(content_item.get("data_type")) %></td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("readable_size")) %></td>'); // size
+        tmpl_array.push('           <% } %>  ');
+        tmpl_array.push('           <td><%= _.escape(content_item.get("time_updated")) %></td>'); // time updated
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('       ');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+
+        tmpl_array.push('</div>');
+        return tmpl_array.join('');
+    },
+    templateDatasetModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<div id="dataset_info_modal">');
+        tmpl_array.push('   <table class="table table-striped table-condensed">');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row" id="id_row" data-id="<%= _.escape(item.get("ldda_id")) %>">Name</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("name")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Data type</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("data_type")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Genome build</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("genome_build")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('           <th scope="row">Size</th>');
+        tmpl_array.push('           <td><%= _.escape(size) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Date uploaded</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("date_uploaded")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Uploaded by</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("uploaded_by")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('           <tr scope="row">');
+        tmpl_array.push('           <th scope="row">Data Lines</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_data_lines")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <th scope="row">Comment Lines</th>');
+        tmpl_array.push('           <% if (item.get("metadata_comment_lines") === "") { %>'); //folder        
+        tmpl_array.push('               <td scope="row"><%= _.escape(item.get("metadata_comment_lines")) %></td>');
+        tmpl_array.push('           <% } else { %>'); 
+        tmpl_array.push('               <td scope="row">unknown</td>');    
+        tmpl_array.push('           <% } %>'); 
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Number of Columns</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_columns")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Column Types</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_column_types")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Miscellaneous information</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("misc_blurb")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('   </table>');
+        tmpl_array.push('   <pre class="peek">');
+        tmpl_array.push('   </pre>');
+        tmpl_array.push('</div>');
+
+        return tmpl_array.join('');
+    },
+
+    templateHistorySelectInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_combo" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_single" name="dataset_import_single" style="width:50%; margin-bottom: 1em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
+
+        return tmpl_array.join('');
+        },   
+
+    templateBulkImportInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_combo_bulk" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_bulk" name="dataset_import_bulk" style="width:50%; margin-bottom: 1em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
+
+        return tmpl_array.join('');
+        },    
+
+      // convert size to nice string
+      size_to_string : function (size)
+      {
+            // identify unit
+            var unit = "";
+            if (size >= 100000000000)   { size = size / 100000000000; unit = "TB"; } else
+            if (size >= 100000000)      { size = size / 100000000; unit = "GB"; } else
+            if (size >= 100000)         { size = size / 100000; unit = "MB"; } else
+            if (size >= 100)            { size = size / 100; unit = "KB"; } else
+            { size = size * 10; unit = "b"; }
+            // return formatted string
+            return (Math.round(size) / 10) + unit;
+        },
+
+// MMMMMMMMMMMMMMM
+// === EVENTS ====
+// MMMMMMMMMMMMMMM
+
+      // event binding
+      events: {
+            'click #select-all-checkboxes' : 'selectAll',
+            'click .folder_row' : 'selectClicked',
+            'click #toolbtn_bulk_import' : 'modalBulkImport',
+            'click #toolbtn_dl' : 'bulkDownload',
+            'click .library-dataset' : 'showDatasetDetails',
+            'click #toolbtn_create_folder' : 'createFolderModal',
+            'click .btn_open_folder' : 'navigateToFolder'
+        },
+
+      //render the folder view
+      render: function (options) {
+        //hack to show scrollbars
+        $("#center").css('overflow','auto');
+
+        view = this;
+        var that = this;
+
+        var folderContainer = new FolderContainer({id: options.id});
+        folderContainer.url = folderContainer.attributes.urlRoot + options.id + '/contents';
+
+        folderContainer.fetch({
+          success: function (container) {
+
+            // prepare nice size strings
+              for (var i = 0; i < folderContainer.attributes.folder.models.length; i++) {
+                  var model = folderContainer.attributes.folder.models[i]
+                  if (model.get('type') === 'file'){
+                    model.set('readable_size', that.size_to_string(model.get('file_size')))
+                  }
+              };
+
+              // find the upper id
+              var path = folderContainer.full_path;
+              var upper_folder_id;
+              if (path.length === 1){ // library is above us
+                upper_folder_id = 0;
+              } else {
+                upper_folder_id = path[path.length-2][0];
+              }
+
+              var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id, upper_folder_id: upper_folder_id });
+              // var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id });
+              that.$el.html(template);
+            }
+          })
+        },
+
+      // handles the click on 'open' and 'upper' folder icons
+      navigateToFolder : function(event){
+        var folder_id = $(event.target).attr('data-id');
+        if (typeof folder_id === 'undefined') {
+            return false;
+        } else if (folder_id === '0'){
+            library_router.navigate('#', {trigger: true, replace: true});
+        } else {
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+        }
+      },
+
+      //show modal with current dataset info
+      showDatasetDetails : function(event){
+        // prevent default
+        event.preventDefault();
+        
+//TODO check whether we already have the data
+
+        //load the ID of the row
+        var id = $(event.target).parent().parent().attr('id');
+        
+        //create new item
+        var item = new Item();
+        var histories = new GalaxyHistories();
+        item.id = id;
+        var self = this;
+
+        //fetch the dataset info
+        item.fetch({
+          success: function (item) {
+// TODO can render here already
+                //fetch user histories for import purposes
+                histories.fetch({
+                    success: function (histories){self.renderModalAfterFetch(item, histories)}
+                });
+            }
+        });
+    },
+
+      // show the current dataset in a modal
+      renderModalAfterFetch : function(item, histories){
+        var size = this.size_to_string(item.get('file_size'));
+        var template = _.template(this.templateDatasetModal(), { item : item, size : size });
+        this.modal = null;
+            // make modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal({
+                title   : 'Dataset Details',
+                body    : template,
+                buttons : {
+                    'Import' : function() { self.importCurrentIntoHistory() },
+                    'Download' : function() { self.downloadCurrent() },
+                    'Close'  : function() { self.modal.hide(); $('.modal').remove(); self.modal = null; } // TODO refill nicely modal with data
+                }
+            });
+            $(".peek").html(item.get("peek"));
+            var history_footer_tmpl = _.template(this.templateHistorySelectInModal(), {histories : histories.models});
+            $(this.modal.elMain).find('.buttons').prepend(history_footer_tmpl);
+
+            // preset last selected history if we know it
+            if (self.lastSelectedHistory.length > 0) {
+                $(this.modal.elMain).find('#dataset_import_single').val(self.lastSelectedHistory);
+            }
+
+            // show the prepared modal
+            this.modal.show();
+        },
+
+        // download dataset shown currently in modal
+        downloadCurrent : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
+
+            var library_dataset_id = [];
+            library_dataset_id.push($('#id_row').attr('data-id'));
+            var url = '/api/libraries/datasets/download/uncompressed';
+            var data = {'ldda_ids' : library_dataset_id};
+
+            // we assume the view is existent
+            folderContentView.processDownload(url, data);
+            this.modal.enableButton('Import');
+            this.modal.enableButton('Download');
+        },
+
+        // import dataset shown currently in modal into selected history
+        importCurrentIntoHistory : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
+
+            var history_id = $(this.modal.elMain).find('select[name=dataset_import_single] option:selected').val();
+            this.lastSelectedHistory = history_id; //save selected history for further use
+
+            var library_dataset_id = $('#id_row').attr('data-id');
+            var historyItem = new HistoryItem();
+            var self = this;
+            historyItem.url = historyItem.urlRoot + history_id + '/contents';
+
+            // save the dataset into selected history
+            historyItem.save({ content : library_dataset_id, source : 'library' }, { success : function(){
+                self.modal.showNotification('Dataset imported', 3000, 'success');
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');
+            }, error : function(){
+                self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, 'error');
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');                
+            }
+        });
+        },
+
+        // select all datasets 
+        selectAll : function (event) {
+           var selected = event.target.checked;
+                 // Iterate each checkbox
+                 $(':checkbox').each(function () { this.checked = selected; });
+                 this.showTools();
+             },
+
+         // click checkbox on folder click
+         selectClicked : function (event) {
+            var checkbox = $("#" + event.target.parentElement.id).find(':checkbox')
+                if (checkbox[0] != undefined) {
+                  if (checkbox[0].checked){
+                    checkbox[0].checked = '';
+                    // $(event.target.parentElement).css('background-color', '').css('color', '');
+                    $(event.target.parentElement).removeClass('dark');
+                    $(event.target.parentElement).find('a').removeClass('dark');
+                    $(event.target.parentElement).addClass('light');
+                    $(event.target.parentElement).find('a').addClass('light');
+                } else {
+                    checkbox[0].checked = 'selected';
+                    $(event.target.parentElement).removeClass('light');
+                    $(event.target.parentElement).find('a').removeClass('light');
+                    $(event.target.parentElement).addClass('dark');
+                    $(event.target.parentElement).find('a').addClass('dark');
+                    // $(event.target.parentElement).css('background-color', '#8389a1').css('color', 'white');
+                }
+            }
+            this.showTools();
+        },
+
+        // show toolbar in case something is selected
+        showTools : function(){
+            var checkedValues = $('#folder_table').find(':checked');
+            if(checkedValues.length > 0){
+                $('#toolbtn_bulk_import').show();
+                $('#toolbtn_dl').show();
+            } else {
+                $('#toolbtn_bulk_import').hide();
+                $('#toolbtn_dl').hide();
+            }
+
+        },
+
+        // show bulk import modal
+        modalBulkImport : function(){
+            var self = this;
+            // fetch histories
+            var histories = new GalaxyHistories();
+            histories.fetch({
+                    success: function (histories){
+                        // make modal
+                        var history_modal_tmpl =  _.template(self.templateBulkImportInModal(), {histories : histories.models});
+                        self.modal = new mod_modal.GalaxyModal({
+                            title   : 'Import into History',
+                            body    : history_modal_tmpl,
+                            buttons : {
+                                'Import' : function() {self.importAllIntoHistory()},
+                                'Close'  : function() {self.modal.hide(); $('.modal').remove(); self.modal = null;}
+                            }
+                        });
+                        // show the prepared modal
+                        self.modal.show();
+                    }
+                });
+        },
+
+        // import all selected datasets into history
+        importAllIntoHistory : function (){
+            //disable the button
+            this.modal.disableButton('Import');
+
+            var history_id = $("select[name=dataset_import_bulk] option:selected").val();
+            var history_name = $("select[name=dataset_import_bulk] option:selected").text();
+
+            var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+            var progress_bar_tmpl = _.template(this.templateProgressBar(), { history_name : history_name });
+            $(this.modal.elMain).find('.modal-body').html(progress_bar_tmpl);
+            
+            // init the progress bar
+            var progressStep = 100 / dataset_ids.length;
+            this.initProgress(progressStep);
+
+            // prepare the dataset objects to be imported
+            var datasets_to_import = [];
+            for (var i = dataset_ids.length - 1; i >= 0; i--) {
+                library_dataset_id = dataset_ids[i];
+                var historyItem = new HistoryItem();
+                var self = this;
+                historyItem.url = historyItem.urlRoot + history_id + '/contents';
+                historyItem.content = library_dataset_id;
+                historyItem.source = 'library';
+                datasets_to_import.push(historyItem);
+            };
+
+            // call the recursive function to call ajax one after each other
+            this.chainCall(datasets_to_import);
+        },
+
+        chainCall: function(history_item_set){
+            var self = this;
+            var popped_item = history_item_set.pop();
+            if (typeof popped_item === "undefined") {
+                self.modal.showNotification('All datasets imported', 3000, 'success');
+                // enable button again
+                self.modal.enableButton('Import');
+                return
+            }
+                var promise = $.when(popped_item.save({content: popped_item.content, source: popped_item.source})).done(function(a1){
+                        self.updateProgress();
+                        responses.push(a1);
+                        self.chainCall(history_item_set);
+                    });
+        },
+
+        initProgress: function(progressStep){
+            this.progress = 0;
+            this.progressStep = progressStep;
+        },
+        updateProgress: function(){
+            this.progress += this.progressStep;
+            $('.progress-bar').width(Math.round(this.progress) + '%');
+            txt_representation = Math.round(this.progress) + '% Complete';
+            $('.completion_span').text(txt_representation);
+        },
+
+      // progress bar 
+      templateProgressBar : function (){
+        var tmpl_array = [];
+
+        tmpl_array.push('<div class="import_text">');
+        tmpl_array.push('Importing selected datasets to history <b><%= _.escape(history_name) %></b>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('<div class="progress">');
+        tmpl_array.push('   <div class="progress-bar" role="progressbar" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100" style="width: 00%;">');
+        tmpl_array.push('       <span class="completion_span">0% Complete</span>');
+        tmpl_array.push('   </div>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('');
+
+        return tmpl_array.join('');
+      },
+
+      // download selected datasets
+      download : function(folder_id, format){
+        var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+
+        var url = '/api/libraries/datasets/download/' + format;
+        var data = {'ldda_ids' : dataset_ids};
+        this.processDownload(url, data);
+      },
+
+      // create hidden form and submit through POST to initialize download
+      processDownload: function(url, data, method){
+        //url and data options required
+        if( url && data ){ 
+                //data can be string of parameters or array/object
+                data = typeof data == 'string' ? data : $.param(data);
+                //split params into form inputs
+                var inputs = '';
+                $.each(data.split('&'), function(){ 
+                        var pair = this.split('=');
+                        inputs+='<input type="hidden" name="'+ pair[0] +'" value="'+ pair[1] +'" />'; 
+                });
+                //send request
+                $('<form action="'+ url +'" method="'+ (method||'post') +'">'+inputs+'</form>')
+                .appendTo('body').submit().remove();
+        };
+      },
+
+      // shows modal for creating folder
+      createFolderModal: function(){
+        alert('creating folder');
+      }
+
+    });
+
+// galaxy library view
+var GalaxyLibraryview = Backbone.View.extend({
+    el: '#center',
+
+    events: {
+        'click #create_new_library_btn' : 'show_library_modal'
+    },
+
+    // initialize
+    initialize : function(){
+    },    
+
+    // template
+    template_library_list : function (){
+        tmpl_array = [];
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; overflow: auto !important; ">');
+
+        tmpl_array.push('');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
+        tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary icon-file ">New Library</a>');
+        tmpl_array.push('<table class="table table-condensed">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('     <th class="button_heading"></th>');
+        tmpl_array.push('     <th>name</th>');
+        tmpl_array.push('     <th>description</th>');
+        tmpl_array.push('     <th>synopsis</th> ');
+        tmpl_array.push('     <th>model type</th> ');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <% _.each(libraries, function(library) { %>');
+        tmpl_array.push('           <tr>');
+        tmpl_array.push('               <td><button title="Open this library" type="button" data-id="<%- library.get("root_folder_id") %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- library.get("name") %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
+        tmpl_array.push('           </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+        
+        tmpl_array.push('</div>');
+        return tmpl_array.join('');
+    },
+
+    // render
+    render: function () {
+        //hack to show scrollbars
+        $("#center").css('overflow','auto');
+        var that = this;
+        // if (typeof libraries === "undefined") {
+          libraries = new Libraries();
+        // }
+        libraries.fetch({
+          success: function (libraries) {
+            var template = _.template(that.template_library_list(), { libraries : libraries.models });
+            that.$el.html(template);
+        }
+    })
+    },
+
+    // own modal
+    modal : null,
+
+    // show/hide create library modal
+    show_library_modal : function (event){
+        event.preventDefault();
+        event.stopPropagation();
+        
+        // create modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal(
+            {
+                title   : 'Create New Library',
+                body    : this.template_new_library(),
+                buttons : {
+                    'Create' : function() {self.create_new_library_event()},
+                    'Close'  : function() {self.modal.hide()}
+                }
+            });
+        
+        // show prepared modal
+        this.modal.show();
+    },
+
+    // create the new library from modal
+    create_new_library_event: function(){
+        var libraryDetails = this.serialize_new_library();
+        var valid = this.validate_new_library(libraryDetails);
+        var library = new Library();
+        var self = this;
+        library.save(libraryDetails, {
+          success: function (library) {
+            self.modal.hide();
+            self.clear_library_modal();
+            self.render();
+          },
+          error: function(){
+            self.modal.showNotification('An error occured', 5000, 'error');
+            }
+    });
+        return false;
+    },
+
+    // clear the library modal once saved
+    clear_library_modal : function(){
+        $("input[name='Name']").val('');
+        $("input[name='Description']").val('');
+        $("input[name='Synopsis']").val('');
+    },
+
+    // serialize data from the form
+    serialize_new_library : function(){
+        return {
+            name: $("input[name='Name']").val(),
+            description: $("input[name='Description']").val(),
+            synopsis: $("input[name='Synopsis']").val()
+        };
+    },
+
+    validate_new_library: function(library){
+
+    },
+
+
+    // template for new library modal
+    template_new_library: function()
+    {
+        tmpl_array = [];
+
+        tmpl_array.push('<div id="new_library_modal">');
+        tmpl_array.push('<form>');
+        tmpl_array.push('<input type="text" name="Name" value="" placeholder="Name">');
+        tmpl_array.push('<input type="text" name="Description" value="" placeholder="Description">');
+        tmpl_array.push('<input type="text" name="Synopsis" value="" placeholder="Synopsis">');
+        tmpl_array.push('</form>');
+        tmpl_array.push('</div>');
+
+        return tmpl_array.join('');
+    }
+});
+
+// galaxy library wrapper View
+var GalaxyLibrary = Backbone.View.extend({
+    folderContentView : null,
+    galaxyLibraryview : null,
+    initialize : function(){
+
+        folderContentView = new FolderContentView();
+        galaxyLibraryview = new GalaxyLibraryview();
+
+        library_router = new LibraryRouter();
+
+        library_router.on('route:libraries', function() {
+          // render libraries list
+          galaxyLibraryview.render();
+      });
+
+        library_router.on('route:folder_content', function(id) {
+          // render folder's contents
+          folderContentView.render({id: id});
+      });    
+
+       library_router.on('route:download', function(folder_id, format) {
+          // send download stream
+          if (typeof folderContentView === 'undefined'){
+            alert('you cant touch this!');
+          // } else if (folderContentView.modal !== null){
+            // folderContentView.download(folder_id, format);
+          } else if ($('#center').find(':checked').length === 0) { // coming from outside of the library app
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+            // TODO show message of redirection
+          } else {
+            folderContentView.download(folder_id, format);
+            library_router.navigate('folders/' + folder_id, {trigger: false, replace: true});
+          }
+
+      });      
+
+Backbone.history.start();   
+
+return this
+}
+});
+
+// return
+return {
+    GalaxyApp: GalaxyLibrary
+};
+
+});
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 static/scripts/galaxy.menu.js
--- a/static/scripts/galaxy.menu.js
+++ b/static/scripts/galaxy.menu.js
@@ -57,7 +57,12 @@
 
         tab_shared.add({
             title   : "Data Libraries",
-            content : "library/index",
+            content : "library/index"
+        });
+
+        tab_shared.add({
+            title   : "New Libraries",
+            content : "library/list",
             divider : true
         });
 
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -1,6 +1,3 @@
-/*
-    galaxy modal
-*/
 
 // dependencies
 define([], function() {
@@ -25,8 +22,51 @@
     
     // initialize
     initialize : function(options) {
+        self = this;
         if (options)
             this.create(options);
+
+        this.bindClick(event, self);
+    },
+
+    // bind the click-to-hide function
+    bindClick: function(event, that) {
+        // bind the ESC key to hide() function
+        $(document).on('keyup', function(event){
+            if (event.keyCode == 27) { self.hide(); }
+        })
+        // bind the click anywhere to hide() function...
+        $('html').on('click', function(event){
+            that.hide();
+        })
+        // ...but don't hide if the click is on modal content
+        $('.modal-content').on('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+    // unbind the click-to-hide function
+    unbindClick: function(event, that){
+        // bind the ESC key to hide() function
+        $(document).off('keyup', function(event){
+            if (event.keyCode == 27) { that.hide(); }
+        })
+        // unbind the click anywhere to hide() function...
+        $('html').off('click', function(event){
+            that.hide();
+        })
+        // ...but don't hide if the click is on modal content
+        $('.modal-content').off('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+
+    // destroy
+    destroy : function(){
+        this.hide();
+        this.unbindClick();
+        $('.modal').remove();
     },
 
     // adds and displays a new frame/window
@@ -87,6 +127,7 @@
         this.$footer  = (this.$el).find('.modal-footer');
         this.$buttons = (this.$el).find('.buttons');
         this.$backdrop = (this.$el).find('.modal-backdrop');
+        this.$notification = (this.$el).find('.notification-modal');
         
         // append body
         this.$body.html(this.options.body);
@@ -120,6 +161,47 @@
         this.$buttons.find('#' + String(name).toLowerCase()).prop('disabled', true);
     },
     
+    // hide buttons
+    hideButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).hide();
+    },
+    // show buttons
+    showButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).show();
+    },
+
+    // show notification
+    showNotification : function(message, duration, type) {
+        // defaults
+        var duration = typeof duration !== 'undefined' ? duration : 1500;
+        // var bgColor = typeof bgColor !== 'undefined' ? bgColor : "#F4E0E1";
+        // var txtColor = typeof txtColor !== 'undefined' ? txtColor : "#A42732";
+        var bgColor;
+        var txtColor;
+
+        if (type === 'error'){
+            bgColor = '#f4e0e1';
+            txtColor = '#a42732';
+        // } else if (type === 'success'){
+        } else { // success is default
+            bgColor = '#e1f4e0';
+            txtColor = '#32a427'; 
+        }
+
+        var HTMLmessage = "<div class='notification-message' style='text-align:center; line-height:16px; '> " + message + " </div>";
+        this.$notification.html("<div id='notification-bar' style='display:none; float: right; height: 16px; width:100%; background-color: " + bgColor + "; z-index: 100; color: " + txtColor + ";border-bottom: 1px solid " + txtColor + ";'>" + HTMLmessage + "</div>");
+
+        var self = this;
+        
+        /*animate the bar*/
+        $('#notification-bar').slideDown(function() {
+            setTimeout(function() {
+                $('#notification-bar').slideUp(function() {self.$notification.html('');});
+            }, duration);
+        });
+
+    },
+
     // returns scroll top for body element
     scrollTop: function()
     {
@@ -139,7 +221,8 @@
                             '<div class="modal-header">' +
                                 '<button type="button" class="close" style="display: none;">×</button>' +
                                 '<h4 class="title">' + title + '</h4>' +
-                            '</div>' +
+                                '<span class="notification-modal"></span>' + 
+                                '</div>' +
                             '<div class="modal-body"></div>' +
                             '<div class="modal-footer">' +
                                 '<div class="buttons" style="float: right;"></div>' +
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 static/scripts/libs/bootstrap.js
--- a/static/scripts/libs/bootstrap.js
+++ b/static/scripts/libs/bootstrap.js
@@ -575,3 +575,158 @@
   }
 
 }(window.jQuery);
+
+/* ========================================================================
+ * Bootstrap: dropdown.js v3.0.0
+ * http://twbs.github.com/bootstrap/javascript.html#dropdowns
+ * ========================================================================
+ * Copyright 2012 Twitter, Inc.
+ *
+ * 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.
+ * ======================================================================== */
+
+
++function ($) { "use strict";
+
+  // DROPDOWN CLASS DEFINITION
+  // =========================
+
+  var backdrop = '.dropdown-backdrop'
+  var toggle   = '[data-toggle=dropdown]'
+  var Dropdown = function (element) {
+    var $el = $(element).on('click.bs.dropdown', this.toggle)
+  }
+
+  Dropdown.prototype.toggle = function (e) {
+    var $this = $(this)
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    clearMenus()
+
+    if (!isActive) {
+      if ('ontouchstart' in document.documentElement && !$parent.closest('.navbar-nav').length) {
+        // if mobile we we use a backdrop because click events don't delegate
+        $('<div class="dropdown-backdrop"/>').insertAfter($(this)).on('click', clearMenus)
+      }
+
+      $parent.trigger(e = $.Event('show.bs.dropdown'))
+
+      if (e.isDefaultPrevented()) return
+
+      $parent
+        .toggleClass('open')
+        .trigger('shown.bs.dropdown')
+
+      $this.focus()
+    }
+
+    return false
+  }
+
+  Dropdown.prototype.keydown = function (e) {
+    if (!/(38|40|27)/.test(e.keyCode)) return
+
+    var $this = $(this)
+
+    e.preventDefault()
+    e.stopPropagation()
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    if (!isActive || (isActive && e.keyCode == 27)) {
+      if (e.which == 27) $parent.find(toggle).focus()
+      return $this.click()
+    }
+
+    var $items = $('[role=menu] li:not(.divider):visible a', $parent)
+
+    if (!$items.length) return
+
+    var index = $items.index($items.filter(':focus'))
+
+    if (e.keyCode == 38 && index > 0)                 index--                        // up
+    if (e.keyCode == 40 && index < $items.length - 1) index++                        // down
+    if (!~index)                                      index=0
+
+    $items.eq(index).focus()
+  }
+
+  function clearMenus() {
+    $(backdrop).remove()
+    $(toggle).each(function (e) {
+      var $parent = getParent($(this))
+      if (!$parent.hasClass('open')) return
+      $parent.trigger(e = $.Event('hide.bs.dropdown'))
+      if (e.isDefaultPrevented()) return
+      $parent.removeClass('open').trigger('hidden.bs.dropdown')
+    })
+  }
+
+  function getParent($this) {
+    var selector = $this.attr('data-target')
+
+    if (!selector) {
+      selector = $this.attr('href')
+      selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
+    }
+
+    var $parent = selector && $(selector)
+
+    return $parent && $parent.length ? $parent : $this.parent()
+  }
+
+
+  // DROPDOWN PLUGIN DEFINITION
+  // ==========================
+
+  var old = $.fn.dropdown
+
+  $.fn.dropdown = function (option) {
+    return this.each(function () {
+      var $this = $(this)
+      var data  = $this.data('dropdown')
+
+      if (!data) $this.data('dropdown', (data = new Dropdown(this)))
+      if (typeof option == 'string') data[option].call($this)
+    })
+  }
+
+  $.fn.dropdown.Constructor = Dropdown
+
+
+  // DROPDOWN NO CONFLICT
+  // ====================
+
+  $.fn.dropdown.noConflict = function () {
+    $.fn.dropdown = old
+    return this
+  }
+
+
+  // APPLY TO STANDARD DROPDOWN ELEMENTS
+  // ===================================
+
+  $(document)
+    .on('click.bs.dropdown.data-api', clearMenus)
+    .on('click.bs.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
+    .on('click.bs.dropdown.data-api'  , toggle, Dropdown.prototype.toggle)
+    .on('keydown.bs.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown)
+
+}(window.jQuery);
\ No newline at end of file
diff -r df88d6121d48b0b193d34445ce49916adf17a2d2 -r 181b293ea65936b7edb338bcbeac3bdea8e53063 static/style/src/less/library.less
--- a/static/style/src/less/library.less
+++ b/static/style/src/less/library.less
@@ -29,6 +29,50 @@
     background-color: @table-bg-accent;
 }
 
+
+tr.light td
+{
+    background-color: white;
+    color: black;
+}
+tr.light:hover td
+{
+    background-color: #f5f5f5;
+    color: #8389a1;
+}
+tr.dark td
+{
+    background-color: #8389a1;
+    color: white;
+}
+tr.dark:hover td
+{
+    background-color: #bbbfd0;
+    color: white;
+}
+a.dark:hover
+{
+    color: yellow;
+    // text-decoration: none;
+}
+a.dark
+{
+    color: white;
+    // text-decoration: none;
+}
+th.button_heading
+{
+    width: 7em;
+}
+div.library_breadcrumb{
+    padding-top: 0.8em;
+    padding-bottom: 0.8em;
+}
+div.library_breadcrumb a:hover{
+    color:green;
+}
+
+
 img.expanderIcon {
     padding-right: 4px;
 }
https://bitbucket.org/galaxy/galaxy-central/commits/17c16125da1a/
Changeset:   17c16125da1a
User:        martenson
Date:        2013-12-11 17:22:38
Summary:     Merged galaxy/galaxy-central into default
Affected #:  4 files
diff -r 181b293ea65936b7edb338bcbeac3bdea8e53063 -r 17c16125da1a00a53a020ff0fd84e5676dbabaf1 lib/galaxy/model/mapping.py
--- a/lib/galaxy/model/mapping.py
+++ b/lib/galaxy/model/mapping.py
@@ -1050,8 +1050,11 @@
     Column( "user_id", Integer, ForeignKey( "galaxy_user.id" ), index=True ),
     Column( "key", TrimmedString( 32 ), index=True, unique=True ) )
 
+
 # With the tables defined we can define the mappers and setup the
 # relationships between the model objects.
+def simple_mapping( model, **kwds ):
+    mapper( model, model.table, properties=kwds )
 
 
 mapper( model.Sample, model.Sample.table,
@@ -1200,64 +1203,62 @@
 
 mapper( model.ValidationError, model.ValidationError.table )
 
-mapper( model.HistoryDatasetAssociation, model.HistoryDatasetAssociation.table,
-    properties=dict(
-        dataset=relation(
-            model.Dataset,
-            primaryjoin=( model.Dataset.table.c.id == model.HistoryDatasetAssociation.table.c.dataset_id ), lazy=False ),
-        # .history defined in History mapper
-        copied_from_history_dataset_association=relation(
-            model.HistoryDatasetAssociation,
-            primaryjoin=( model.HistoryDatasetAssociation.table.c.copied_from_history_dataset_association_id == model.HistoryDatasetAssociation.table.c.id ),
-            remote_side=[model.HistoryDatasetAssociation.table.c.id],
-            uselist=False ),
-        copied_to_history_dataset_associations=relation(
-            model.HistoryDatasetAssociation,
-            primaryjoin=( model.HistoryDatasetAssociation.table.c.copied_from_history_dataset_association_id == model.HistoryDatasetAssociation.table.c.id ) ),
-        copied_from_library_dataset_dataset_association=relation(
-            model.LibraryDatasetDatasetAssociation,
-            primaryjoin=( model.HistoryDatasetAssociation.table.c.copied_from_library_dataset_dataset_association_id == model.LibraryDatasetDatasetAssociation.table.c.id ),
-            uselist=False ),
-        copied_to_library_dataset_dataset_associations=relation(
-            model.LibraryDatasetDatasetAssociation,
-            primaryjoin=( model.HistoryDatasetAssociation.table.c.copied_from_library_dataset_dataset_association_id == model.LibraryDatasetDatasetAssociation.table.c.id ) ),
-        implicitly_converted_datasets=relation(
-            model.ImplicitlyConvertedDatasetAssociation,
-            primaryjoin=( model.ImplicitlyConvertedDatasetAssociation.table.c.hda_parent_id == model.HistoryDatasetAssociation.table.c.id ) ),
-        implicitly_converted_parent_datasets=relation(
-            model.ImplicitlyConvertedDatasetAssociation,
-            primaryjoin=( model.ImplicitlyConvertedDatasetAssociation.table.c.hda_id == model.HistoryDatasetAssociation.table.c.id ) ),
-        children=relation(
-            model.HistoryDatasetAssociation,
-            primaryjoin=( model.HistoryDatasetAssociation.table.c.parent_id == model.HistoryDatasetAssociation.table.c.id ),
-            backref=backref( "parent", primaryjoin=( model.HistoryDatasetAssociation.table.c.parent_id == model.HistoryDatasetAssociation.table.c.id ), remote_side=[model.HistoryDatasetAssociation.table.c.id], uselist=False ) ),
-        visible_children=relation(
-            model.HistoryDatasetAssociation,
-            primaryjoin=( ( model.HistoryDatasetAssociation.table.c.parent_id == model.HistoryDatasetAssociation.table.c.id ) & ( model.HistoryDatasetAssociation.table.c.visible == True ) ) ),
-        tags=relation( model.HistoryDatasetAssociationTagAssociation, order_by=model.HistoryDatasetAssociationTagAssociation.table.c.id, backref='history_tag_associations' ),
-        annotations=relation( model.HistoryDatasetAssociationAnnotationAssociation, order_by=model.HistoryDatasetAssociationAnnotationAssociation.table.c.id, backref="hdas" ),
-        ratings=relation( model.HistoryDatasetAssociationRatingAssociation, order_by=model.HistoryDatasetAssociationRatingAssociation.table.c.id, backref="hdas" ) )
-            )
+simple_mapping( model.HistoryDatasetAssociation,
+    dataset=relation(
+        model.Dataset,
+        primaryjoin=( model.Dataset.table.c.id == model.HistoryDatasetAssociation.table.c.dataset_id ), lazy=False ),
+    # .history defined in History mapper
+    copied_from_history_dataset_association=relation(
+        model.HistoryDatasetAssociation,
+        primaryjoin=( model.HistoryDatasetAssociation.table.c.copied_from_history_dataset_association_id == model.HistoryDatasetAssociation.table.c.id ),
+        remote_side=[model.HistoryDatasetAssociation.table.c.id],
+        uselist=False ),
+    copied_to_history_dataset_associations=relation(
+        model.HistoryDatasetAssociation,
+        primaryjoin=( model.HistoryDatasetAssociation.table.c.copied_from_history_dataset_association_id == model.HistoryDatasetAssociation.table.c.id ) ),
+    copied_from_library_dataset_dataset_association=relation(
+        model.LibraryDatasetDatasetAssociation,
+        primaryjoin=( model.HistoryDatasetAssociation.table.c.copied_from_library_dataset_dataset_association_id == model.LibraryDatasetDatasetAssociation.table.c.id ),
+        uselist=False ),
+    copied_to_library_dataset_dataset_associations=relation(
+        model.LibraryDatasetDatasetAssociation,
+        primaryjoin=( model.HistoryDatasetAssociation.table.c.copied_from_library_dataset_dataset_association_id == model.LibraryDatasetDatasetAssociation.table.c.id ) ),
+    implicitly_converted_datasets=relation(
+        model.ImplicitlyConvertedDatasetAssociation,
+        primaryjoin=( model.ImplicitlyConvertedDatasetAssociation.table.c.hda_parent_id == model.HistoryDatasetAssociation.table.c.id ) ),
+    implicitly_converted_parent_datasets=relation(
+        model.ImplicitlyConvertedDatasetAssociation,
+        primaryjoin=( model.ImplicitlyConvertedDatasetAssociation.table.c.hda_id == model.HistoryDatasetAssociation.table.c.id ) ),
+    children=relation(
+        model.HistoryDatasetAssociation,
+        primaryjoin=( model.HistoryDatasetAssociation.table.c.parent_id == model.HistoryDatasetAssociation.table.c.id ),
+        backref=backref( "parent", primaryjoin=( model.HistoryDatasetAssociation.table.c.parent_id == model.HistoryDatasetAssociation.table.c.id ), remote_side=[model.HistoryDatasetAssociation.table.c.id], uselist=False ) ),
+    visible_children=relation(
+        model.HistoryDatasetAssociation,
+        primaryjoin=( ( model.HistoryDatasetAssociation.table.c.parent_id == model.HistoryDatasetAssociation.table.c.id ) & ( model.HistoryDatasetAssociation.table.c.visible == True ) ) ),
+    tags=relation( model.HistoryDatasetAssociationTagAssociation, order_by=model.HistoryDatasetAssociationTagAssociation.table.c.id, backref='history_tag_associations' ),
+    annotations=relation( model.HistoryDatasetAssociationAnnotationAssociation, order_by=model.HistoryDatasetAssociationAnnotationAssociation.table.c.id, backref="hdas" ),
+    ratings=relation( model.HistoryDatasetAssociationRatingAssociation, order_by=model.HistoryDatasetAssociationRatingAssociation.table.c.id, backref="hdas" )
+)
 
-mapper( model.Dataset, model.Dataset.table,
-    properties=dict(
-        history_associations=relation(
-            model.HistoryDatasetAssociation,
-            primaryjoin=( model.Dataset.table.c.id == model.HistoryDatasetAssociation.table.c.dataset_id ) ),
-        active_history_associations=relation(
-            model.HistoryDatasetAssociation,
-            primaryjoin=( ( model.Dataset.table.c.id == model.HistoryDatasetAssociation.table.c.dataset_id ) & ( model.HistoryDatasetAssociation.table.c.deleted == False ) & ( model.HistoryDatasetAssociation.table.c.purged == False ) ) ),
-        purged_history_associations=relation(
-            model.HistoryDatasetAssociation,
-            primaryjoin=( ( model.Dataset.table.c.id == model.HistoryDatasetAssociation.table.c.dataset_id ) & ( model.HistoryDatasetAssociation.table.c.purged == True ) ) ),
-        library_associations=relation(
-            model.LibraryDatasetDatasetAssociation,
-            primaryjoin=( model.Dataset.table.c.id == model.LibraryDatasetDatasetAssociation.table.c.dataset_id ) ),
-        active_library_associations=relation(
-            model.LibraryDatasetDatasetAssociation,
-            primaryjoin=( ( model.Dataset.table.c.id == model.LibraryDatasetDatasetAssociation.table.c.dataset_id ) & ( model.LibraryDatasetDatasetAssociation.table.c.deleted == False ) ) ),
-        tags=relation(model.DatasetTagAssociation, order_by=model.DatasetTagAssociation.table.c.id, backref='datasets')
-            ) )
+simple_mapping( model.Dataset,
+    history_associations=relation(
+        model.HistoryDatasetAssociation,
+        primaryjoin=( model.Dataset.table.c.id == model.HistoryDatasetAssociation.table.c.dataset_id ) ),
+    active_history_associations=relation(
+        model.HistoryDatasetAssociation,
+        primaryjoin=( ( model.Dataset.table.c.id == model.HistoryDatasetAssociation.table.c.dataset_id ) & ( model.HistoryDatasetAssociation.table.c.deleted == False ) & ( model.HistoryDatasetAssociation.table.c.purged == False ) ) ),
+    purged_history_associations=relation(
+        model.HistoryDatasetAssociation,
+        primaryjoin=( ( model.Dataset.table.c.id == model.HistoryDatasetAssociation.table.c.dataset_id ) & ( model.HistoryDatasetAssociation.table.c.purged == True ) ) ),
+    library_associations=relation(
+        model.LibraryDatasetDatasetAssociation,
+        primaryjoin=( model.Dataset.table.c.id == model.LibraryDatasetDatasetAssociation.table.c.dataset_id ) ),
+    active_library_associations=relation(
+        model.LibraryDatasetDatasetAssociation,
+        primaryjoin=( ( model.Dataset.table.c.id == model.LibraryDatasetDatasetAssociation.table.c.dataset_id ) & ( model.LibraryDatasetDatasetAssociation.table.c.deleted == False ) ) ),
+    tags=relation(model.DatasetTagAssociation, order_by=model.DatasetTagAssociation.table.c.id, backref='datasets')
+)
 
 mapper( model.HistoryDatasetAssociationDisplayAtAuthorization, model.HistoryDatasetAssociationDisplayAtAuthorization.table,
     properties=dict( history_dataset_association = relation( model.HistoryDatasetAssociation ),
@@ -1751,90 +1752,63 @@
                  ) )
 
 # Tag tables.
+simple_mapping( model.Tag,
+    children=relation(model.Tag, backref=backref( 'parent', remote_side=[model.Tag.table.c.id] ) )
+)
 
-mapper( model.Tag, model.Tag.table,
-    properties=dict( children=relation(model.Tag, backref=backref( 'parent', remote_side=[model.Tag.table.c.id] ) )
-                     ) )
 
-mapper( model.HistoryTagAssociation, model.HistoryTagAssociation.table,
-    properties=dict( tag=relation(model.Tag, backref="tagged_histories"), user=relation( model.User ) )
-                     )
+def tag_mapping( tag_association_class, backref_name ):
+    simple_mapping( tag_association_class, tag=relation( model.Tag, backref=backref_name), user=relation( model.User ) )
 
-mapper( model.DatasetTagAssociation, model.DatasetTagAssociation.table,
-    properties=dict( tag=relation(model.Tag, backref="tagged_datasets"), user=relation( model.User ) )
-                     )
+tag_mapping( model.HistoryTagAssociation, "tagged_histories" )
 
-mapper( model.HistoryDatasetAssociationTagAssociation, model.HistoryDatasetAssociationTagAssociation.table,
-    properties=dict( tag=relation(model.Tag, backref="tagged_history_dataset_associations"), user=relation( model.User ) )
-                     )
+tag_mapping( model.DatasetTagAssociation, "tagged_datasets" )
 
-mapper( model.PageTagAssociation, model.PageTagAssociation.table,
-    properties=dict( tag=relation(model.Tag, backref="tagged_pages"), user=relation( model.User ) )
-                    )
+tag_mapping( model.HistoryDatasetAssociationTagAssociation, "tagged_history_dataset_associations" )
 
-mapper( model.StoredWorkflowTagAssociation, model.StoredWorkflowTagAssociation.table,
-    properties=dict( tag=relation(model.Tag, backref="tagged_workflows"), user=relation( model.User ) )
-                    )
+tag_mapping( model.PageTagAssociation, "tagged_pages" )
 
-mapper( model.WorkflowStepTagAssociation, model.WorkflowStepTagAssociation.table,
-    properties=dict( tag=relation(model.Tag, backref="tagged_workflow_steps"), user=relation( model.User ) )
-                    )
+tag_mapping( model.StoredWorkflowTagAssociation, "tagged_workflows" )
 
-mapper( model.VisualizationTagAssociation, model.VisualizationTagAssociation.table,
-    properties=dict( tag=relation(model.Tag, backref="tagged_visualizations"), user=relation( model.User ) )
-                    )
+tag_mapping( model.WorkflowStepTagAssociation, "tagged_workflow_steps" )
 
-mapper( model.ToolTagAssociation, model.ToolTagAssociation.table,
-    properties=dict( tag=relation(model.Tag, backref="tagged_tools"), user=relation( model.User ) )
-                    )
+tag_mapping( model.VisualizationTagAssociation, "tagged_visualizations" )
+
+tag_mapping( model.ToolTagAssociation, "tagged_tools" )
+
 
 # Annotation tables.
+def annotation_mapping( annotation_class, **kwds ):
+    kwds = dict( [ (key, relation( value ) ) for key, value in kwds.iteritems() ] )
+    simple_mapping( annotation_class, **dict(user=relation( model.User ), **kwds ) )
 
-mapper( model.HistoryAnnotationAssociation, model.HistoryAnnotationAssociation.table,
-    properties=dict( history=relation( model.History ), user=relation( model.User ) )
-                    )
+annotation_mapping( model.HistoryAnnotationAssociation, history=model.History )
 
-mapper( model.HistoryDatasetAssociationAnnotationAssociation, model.HistoryDatasetAssociationAnnotationAssociation.table,
-    properties=dict( hda=relation( model.HistoryDatasetAssociation ), user=relation( model.User ) )
-                    )
+annotation_mapping( model.HistoryDatasetAssociationAnnotationAssociation, hda=model.HistoryDatasetAssociation )
 
-mapper( model.StoredWorkflowAnnotationAssociation, model.StoredWorkflowAnnotationAssociation.table,
-    properties=dict( stored_workflow=relation( model.StoredWorkflow ), user=relation( model.User ) )
-                    )
+annotation_mapping( model.StoredWorkflowAnnotationAssociation, stored_workflow=model.StoredWorkflow )
 
-mapper( model.WorkflowStepAnnotationAssociation, model.WorkflowStepAnnotationAssociation.table,
-    properties=dict( workflow_step=relation( model.WorkflowStep ), user=relation( model.User ) )
-                    )
+annotation_mapping( model.WorkflowStepAnnotationAssociation, workflow_step=model.WorkflowStep )
 
-mapper( model.PageAnnotationAssociation, model.PageAnnotationAssociation.table,
-    properties=dict( page=relation( model.Page ), user=relation( model.User ) )
-                    )
+annotation_mapping( model.PageAnnotationAssociation, page=model.Page )
 
-mapper( model.VisualizationAnnotationAssociation, model.VisualizationAnnotationAssociation.table,
-    properties=dict( visualization=relation( model.Visualization ), user=relation( model.User ) )
-                    )
+annotation_mapping( model.VisualizationAnnotationAssociation, visualization=model.Visualization )
+
 
 # Rating tables.
+def rating_mapping( rating_class, **kwds ):
+    kwds = dict( [ (key, relation( value ) ) for key, value in kwds.iteritems() ] )
+    simple_mapping( rating_class, **dict(user=relation( model.User ), **kwds ) )
 
-mapper( model.HistoryRatingAssociation, model.HistoryRatingAssociation.table,
-    properties=dict( history=relation( model.History ), user=relation( model.User ) )
-                    )
+rating_mapping( model.HistoryRatingAssociation, history=model.History )
 
-mapper( model.HistoryDatasetAssociationRatingAssociation, model.HistoryDatasetAssociationRatingAssociation.table,
-    properties=dict( hda=relation( model.HistoryDatasetAssociation ), user=relation( model.User ) )
-                    )
+rating_mapping( model.HistoryDatasetAssociationRatingAssociation, hda=model.HistoryDatasetAssociation )
 
-mapper( model.StoredWorkflowRatingAssociation, model.StoredWorkflowRatingAssociation.table,
-    properties=dict( stored_workflow=relation( model.StoredWorkflow ), user=relation( model.User ) )
-                    )
+rating_mapping( model.StoredWorkflowRatingAssociation, stored_workflow=model.StoredWorkflow )
 
-mapper( model.PageRatingAssociation, model.PageRatingAssociation.table,
-    properties=dict( page=relation( model.Page ), user=relation( model.User ) )
-                    )
+rating_mapping( model.PageRatingAssociation, page=model.Page )
 
-mapper( model.VisualizationRatingAssociation, model.VisualizationRatingAssociation.table,
-    properties=dict( visualization=relation( model.Visualization ), user=relation( model.User ) )
-                    )
+rating_mapping( model.VisualizationRatingAssociation, visualizaiton=model.Visualization )
 
 #Data Manager tables
 mapper( model.DataManagerHistoryAssociation, model.DataManagerHistoryAssociation.table,
diff -r 181b293ea65936b7edb338bcbeac3bdea8e53063 -r 17c16125da1a00a53a020ff0fd84e5676dbabaf1 lib/galaxy/tools/__init__.py
--- a/lib/galaxy/tools/__init__.py
+++ b/lib/galaxy/tools/__init__.py
@@ -3003,10 +3003,10 @@
         # Add link details.
         if link_details:
             # Add details for creating a hyperlink to the tool.
-            if not self.tool_type.startswith( 'data_source' ):
-                link = url_for( '/tool_runner', tool_id=self.id )
+            if not isinstance( self, DataSourceTool ):
+                link = url_for( controller='tool_runner', tool_id=self.id )
             else:
-                link = url_for( self.action, **self.get_static_param_values( trans ) )
+                link = url_for( controller='tool_runner', action='data_source_redirect', tool_id=self.id )
 
             # Basic information
             tool_dict.update( { 'link': link,
diff -r 181b293ea65936b7edb338bcbeac3bdea8e53063 -r 17c16125da1a00a53a020ff0fd84e5676dbabaf1 lib/galaxy/webapps/galaxy/controllers/tool_runner.py
--- a/lib/galaxy/webapps/galaxy/controllers/tool_runner.py
+++ b/lib/galaxy/webapps/galaxy/controllers/tool_runner.py
@@ -6,6 +6,7 @@
 import galaxy.util
 from galaxy import web
 from galaxy.tools import DefaultToolState
+from galaxy.tools import DataSourceTool
 from galaxy.tools.actions import upload_common
 from galaxy.tools.parameters import params_to_incoming
 from galaxy.tools.parameters import visit_input_values
@@ -233,6 +234,35 @@
                                     **vars )
 
     @web.expose
+    def data_source_redirect( self, trans, tool_id=None ):
+        """
+        Redirects a user accessing a Data Source tool to its target action link.
+        This method will subvert mix-mode content blocking in several browsers when
+        accessing non-https data_source tools from an https galaxy server.
+        
+        Tested as working on Safari 7.0 and FireFox 26
+        Subverting did not work on Chrome 31
+        """
+        if tool_id is None:
+            return trans.response.send_redirect( url_for( controller="root", action="welcome" ) )
+        tool_version_select_field, tools, tool = self.__get_tool_components( tool_id,
+                                                                             tool_version=None,
+                                                                             get_loaded_tools_by_lineage=False,
+                                                                             set_selected=False )
+        # No tool matching the tool id, display an error (shouldn't happen)
+        if not tool:
+            log.error( "data_source_redirect called with tool id '%s' but no such tool exists", tool_id )
+            trans.log_event( "Tool id '%s' does not exist" % tool_id )
+            trans.response.status = 404
+            return "Tool '%s' does not exist, kwd=%s " % ( tool_id, kwd )
+        
+        if isinstance( tool, DataSourceTool ):
+            link = url_for( tool.action, **tool.get_static_param_values( trans ) )
+        else:
+            link = url_for( controller='tool_runner', tool_id=tool.id )
+        return trans.response.send_redirect( link )
+    
+    @web.expose
     def redirect( self, trans, redirect_url=None, **kwd ):
         if not redirect_url:
             return trans.show_error_message( "Required URL for redirection missing" )
diff -r 181b293ea65936b7edb338bcbeac3bdea8e53063 -r 17c16125da1a00a53a020ff0fd84e5676dbabaf1 test/unit/test_galaxy_mapping.py
--- a/test/unit/test_galaxy_mapping.py
+++ b/test/unit/test_galaxy_mapping.py
@@ -3,10 +3,139 @@
 
 
 class MappingTests( unittest.TestCase ):
+
+    def test_annotations( self ):
+        model = self.model
+
+        u = model.User( email="annotator(a)example.com", password="password" )
+        self.persist( u )
+
+        def persist_and_check_annotation( annotation_class, **kwds ):
+            annotated_association = annotation_class()
+            annotated_association.annotation = "Test Annotation"
+            annotated_association.user = u
+            for key, value in kwds.iteritems():
+                setattr(annotated_association, key, value)
+            self.persist( annotated_association )
+            self.expunge()
+            stored_annotation = self.query( annotation_class ).all()[0]
+            assert stored_annotation.annotation == "Test Annotation"
+            assert stored_annotation.user.email == "annotator(a)example.com"
+
+        sw = model.StoredWorkflow()
+        sw.user = u
+        self.persist( sw )
+        persist_and_check_annotation( model.StoredWorkflowAnnotationAssociation, stored_workflow=sw )
+
+        workflow = model.Workflow()
+        workflow.stored_workflow = sw
+        self.persist( workflow )
+
+        ws = model.WorkflowStep()
+        ws.workflow = workflow
+        self.persist( ws )
+        persist_and_check_annotation( model.WorkflowStepAnnotationAssociation, workflow_step=ws )
+
+        h = model.History( name="History for Annotation", user=u)
+        self.persist( h )
+        persist_and_check_annotation( model.HistoryAnnotationAssociation, history=h )
+
+        d1 = model.HistoryDatasetAssociation( extension="txt", history=h, create_dataset=True, sa_session=model.session )
+        self.persist( d1 )
+        persist_and_check_annotation( model.HistoryDatasetAssociationAnnotationAssociation, hda=d1 )
+
+        page = model.Page()
+        page.user = u
+        self.persist( page )
+        persist_and_check_annotation( model.PageAnnotationAssociation, page=page )
+
+        visualization = model.Visualization()
+        visualization.user = u
+        self.persist( visualization )
+        persist_and_check_annotation( model.VisualizationAnnotationAssociation, visualization=visualization )
+
+    def test_ratings( self ):
+        model = self.model
+
+        u = model.User( email="rater(a)example.com", password="password" )
+        self.persist( u )
+
+        def persist_and_check_rating( rating_class, **kwds ):
+            rating_association = rating_class()
+            rating_association.rating = 5
+            rating_association.user = u
+            for key, value in kwds.iteritems():
+                setattr(rating_association, key, value)
+            self.persist( rating_association )
+            self.expunge()
+            stored_annotation = self.query( rating_class ).all()[0]
+            assert stored_annotation.rating == 5
+            assert stored_annotation.user.email == "rater(a)example.com"
+
+        sw = model.StoredWorkflow()
+        sw.user = u
+        self.persist( sw )
+        persist_and_check_rating( model.StoredWorkflowRatingAssociation, stored_workflow=sw )
+
+        h = model.History( name="History for Rating", user=u)
+        self.persist( h )
+        persist_and_check_rating( model.HistoryRatingAssociation, history=h )
+
+        d1 = model.HistoryDatasetAssociation( extension="txt", history=h, create_dataset=True, sa_session=model.session )
+        self.persist( d1 )
+        persist_and_check_rating( model.HistoryDatasetAssociationRatingAssociation, hda=d1 )
+
+        page = model.Page()
+        page.user = u
+        self.persist( page )
+        persist_and_check_rating( model.PageRatingAssociation, page=page )
+
+        visualization = model.Visualization()
+        visualization.user = u
+        self.persist( visualization )
+        persist_and_check_rating( model.VisualizationRatingAssociation, visualization=visualization )
+
+    def test_tags( self ):
+        model = self.model
+
+        my_tag = model.Tag(name="Test Tag")
+        u = model.User( email="tagger(a)example.com", password="password" )
+        self.persist( my_tag, u )
+
+        def tag_and_test( taggable_object, tag_association_class, backref_name ):
+            assert len( getattr(self.query( model.Tag ).filter( model.Tag.name == "Test Tag" ).all()[0], backref_name) ) == 0
+
+            tag_association = tag_association_class()
+            tag_association.tag = my_tag
+            taggable_object.tags = [ tag_association ]
+            self.persist( tag_association, taggable_object )
+
+            assert len( getattr(self.query( model.Tag ).filter( model.Tag.name == "Test Tag" ).all()[0], backref_name) ) == 1
+
+        sw = model.StoredWorkflow()
+        sw.user = u
+        #self.persist( sw )
+        tag_and_test( sw, model.StoredWorkflowTagAssociation, "tagged_workflows" )
+
+        h = model.History( name="History for Tagging", user=u)
+        tag_and_test( h, model.HistoryTagAssociation, "tagged_histories" )
+
+        d1 = model.HistoryDatasetAssociation( extension="txt", history=h, create_dataset=True, sa_session=model.session )
+        tag_and_test( d1, model.HistoryDatasetAssociationTagAssociation, "tagged_history_dataset_associations" )
+
+        page = model.Page()
+        page.user = u
+        tag_and_test( page, model.PageTagAssociation, "tagged_pages" )
+
+        visualization = model.Visualization()
+        visualization.user = u
+        tag_and_test( visualization, model.VisualizationTagAssociation, "tagged_visualizations" )
+
     def test_basic( self ):
-        # Start the database and connect the mapping
-        model = mapping.init( "/tmp", "sqlite:///:memory:", create_tables=True )
-        assert model.engine is not None
+        model = self.model
+
+        original_user_count = len( model.session.query( model.User ).all() )
+
         # Make some changes and commit them
         u = model.User( email="james(a)foo.bar.baz", password="password" )
         # gs = model.GalaxySession()
@@ -14,40 +143,64 @@
         #h1.queries.append( model.Query( "h1->q1" ) )
         #h1.queries.append( model.Query( "h1->q2" ) )
         h2 = model.History( name=( "H" * 1024 ) )
-        model.session.add_all( ( u, h1, h2 ) )
+        self.persist( u, h1, h2 )
         #q1 = model.Query( "h2->q1" )
         metadata = dict( chromCol=1, startCol=2, endCol=3 )
         d1 = model.HistoryDatasetAssociation( extension="interval", metadata=metadata, history=h2, create_dataset=True, sa_session=model.session )
         #h2.queries.append( q1 )
         #h2.queries.append( model.Query( "h2->q2" ) )
-        model.session.add( ( d1 ) )
-        model.session.flush()
-        model.session.expunge_all()
+        self.persist( d1 )
+
         # Check
         users = model.session.query( model.User ).all()
-        assert len( users ) == 1
-        assert users[0].email == "james(a)foo.bar.baz"
-        assert users[0].password == "password"
-        assert len( users[0].histories ) == 1
-        assert users[0].histories[0].name == "History 1"
+        assert len( users ) == original_user_count + 1
+        user = [user for user in users if user.email == "james(a)foo.bar.baz"][0]
+        assert user.email == "james(a)foo.bar.baz"
+        assert user.password == "password"
+        assert len( user.histories ) == 1
+        assert user.histories[0].name == "History 1"
         hists = model.session.query( model.History ).all()
-        assert hists[0].name == "History 1"
-        assert hists[1].name == ( "H" * 255 )
-        assert hists[0].user == users[0]
-        assert hists[1].user is None
-        assert hists[1].datasets[0].metadata.chromCol == 1
+        hist0 = [history for history in hists if history.name == "History 1"][0]
+        hist1 = [history for history in hists if history.name == "H" * 255][0]
+        assert hist0.name == "History 1"
+        assert hist1.name == ( "H" * 255 )
+        assert hist0.user == user
+        assert hist1.user is None
+        assert hist1.datasets[0].metadata.chromCol == 1
         # The filename test has moved to objecstore
-        #id = hists[1].datasets[0].id
-        #assert hists[1].datasets[0].file_name == os.path.join( "/tmp", *directory_hash_id( id ) ) + ( "/dataset_%d.dat" % id )
+        #id = hist1.datasets[0].id
+        #assert hist1.datasets[0].file_name == os.path.join( "/tmp", *directory_hash_id( id ) ) + ( "/dataset_%d.dat" % id )
         # Do an update and check
-        hists[1].name = "History 2b"
-        model.session.flush()
-        model.session.expunge_all()
+        hist1.name = "History 2b"
+        self.expunge()
         hists = model.session.query( model.History ).all()
-        assert hists[0].name == "History 1"
-        assert hists[1].name == "History 2b"
+        hist0 = [history for history in hists if history.name == "History 1"][0]
+        hist1 = [history for history in hists if history.name == "History 2b"][0]
+        assert hist0.name == "History 1"
+        assert hist1.name == "History 2b"
         # gvk TODO need to ad test for GalaxySessions, but not yet sure what they should look like.
 
+    @classmethod
+    def setUpClass(cls):
+        # Start the database and connect the mapping
+        cls.model = mapping.init( "/tmp", "sqlite:///:memory:", create_tables=True )
+        assert cls.model.engine is not None
+
+    @classmethod
+    def query( cls, type ):
+        return cls.model.session.query( type )
+
+    @classmethod
+    def persist(cls, *args):
+        for arg in args:
+            cls.model.session.add( arg )
+        cls.expunge()
+
+    @classmethod
+    def expunge(cls):
+        cls.model.session.flush()
+        cls.model.session.expunge_all()
+
 
 def get_suite():
     suite = unittest.TestSuite()
https://bitbucket.org/galaxy/galaxy-central/commits/ff102054eec1/
Changeset:   ff102054eec1
User:        martenson
Date:        2013-12-11 19:03:02
Summary:     merge into default
Affected #:  17 files
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 lib/galaxy/model/__init__.py
--- a/lib/galaxy/model/__init__.py
+++ b/lib/galaxy/model/__init__.py
@@ -1827,7 +1827,7 @@
 class Library( object, Dictifiable ):
     permitted_actions = get_permitted_actions( filter='LIBRARY' )
     dict_collection_visible_keys = ( 'id', 'name' )
-    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis' )
+    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis', 'root_folder_id' )
     def __init__( self, name=None, description=None, synopsis=None, root_folder=None ):
         self.name = name or "Unnamed library"
         self.description = description
@@ -1894,7 +1894,7 @@
         return name
 
 class LibraryFolder( object, Dictifiable ):
-    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build' )
+    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build', 'update_time' )
     def __init__( self, name=None, description=None, item_count=0, order_id=None ):
         self.name = name or "Unnamed folder"
         self.description = description
@@ -2060,6 +2060,7 @@
                      genome_build = ldda.dbkey,
                      misc_info = ldda.info,
                      misc_blurb = ldda.blurb,
+                     peek = ( lambda ldda: ldda.display_peek() if ldda.peek and ldda.peek != 'no peek' else None )( ldda ),
                      template_data = template_data )
         if ldda.dataset.uuid is None:
             rval['uuid'] = None
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 lib/galaxy/model/mapping.py
--- a/lib/galaxy/model/mapping.py
+++ b/lib/galaxy/model/mapping.py
@@ -1852,8 +1852,9 @@
     table = self.table
     trans = conn.begin()
     try:
-        next_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
-        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid + 1 ) )
+        current_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
+        next_hid = current_hid + 1
+        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid ) )
         trans.commit()
         return next_hid
     except:
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 lib/galaxy/webapps/galaxy/api/datasets.py
--- a/lib/galaxy/webapps/galaxy/api/datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/datasets.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a dataset.
+API operations on the contents of a history dataset.
 """
 from galaxy import web
 from galaxy.visualization.data_providers.genome import FeatureLocationIndexDataProvider
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a library.
+API operations on the contents of a folder.
 """
 import logging, os, string, shutil, urllib, re, socket
 from cgi import escape, FieldStorage
@@ -11,67 +11,122 @@
 log = logging.getLogger( __name__ )
 
 class FolderContentsController( BaseAPIController, UsesLibraryMixin, UsesLibraryMixinItems ):
+    """
+    Class controls retrieval, creation and updating of folder contents.
+    """
+
+    def load_folder_contents( self, trans, folder ):
+        """
+        Loads all contents of the folder (folders and data sets) but only in the first level.
+        """
+        current_user_roles = trans.get_current_user_roles()
+        is_admin = trans.user_is_admin()
+        content_items = []
+        for subfolder in folder.active_folders:
+            if not is_admin:
+                can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
+            if (is_admin or can_access) and not subfolder.deleted:
+                subfolder.api_type = 'folder'
+                content_items.append( subfolder )
+        for dataset in folder.datasets:
+            if not is_admin:
+                can_access = trans.app.security_agent.can_access_dataset( current_user_roles, dataset.library_dataset_dataset_association.dataset )
+            if (is_admin or can_access) and not dataset.deleted:
+                dataset.api_type = 'file'
+                content_items.append( dataset )
+        return content_items
 
     @web.expose_api
     def index( self, trans, folder_id, **kwd ):
         """
         GET /api/folders/{encoded_folder_id}/contents
         Displays a collection (list) of a folder's contents (files and folders).
-        The /api/library_contents/{encoded_library_id}/contents
-        lists everything in a library recursively, which is not what
-        we want here. We could add a parameter to use the recursive
-        style, but this is meant to act similar to an "ls" directory listing.
+        Encoded folder ID is prepended with 'F' if it is a folder as opposed to a data set which does not have it.
+        Full path is provided as a separate object in response providing data for breadcrumb path building.
         """
-        rval = []
+        folder_container = []
         current_user_roles = trans.get_current_user_roles()
 
-        def traverse( folder ):
-            admin = trans.user_is_admin()
-            rval = []
-            for subfolder in folder.active_folders:
-                if not admin:
-                    can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
-                if (admin or can_access) and not subfolder.deleted:
-                    subfolder.api_type = 'folder'
-                    rval.append( subfolder )
-            for ld in folder.datasets:
-                if not admin:
-                    can_access = trans.app.security_agent.can_access_dataset( current_user_roles, ld.library_dataset_dataset_association.dataset )
-                if (admin or can_access) and not ld.deleted:
-                    ld.api_type = 'file'
-                    rval.append( ld )
-            return rval
-
-        try:
-            decoded_folder_id = trans.security.decode_id( folder_id[-16:] )
-        except TypeError:
-            trans.response.status = 400
-            return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
+        if ( folder_id.startswith( 'F' ) ):
+            try:
+                decoded_folder_id = trans.security.decode_id( folder_id[1:] )
+            except TypeError:
+                trans.response.status = 400
+                return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
 
         try:
             folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( decoded_folder_id )
-            parent_library = folder.parent_library
         except:
             folder = None
-            log.error( "FolderContentsController.index: Unable to retrieve folder %s"
-                      % folder_id )
+            log.error( "FolderContentsController.index: Unable to retrieve folder with ID: %s" % folder_id )
 
-        # TODO: Find the API's path to this folder if necessary.
-        # This was needed in recursive descent, but it's not needed
-        # for "ls"-style content checking:
-        if not folder or not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+        # We didn't find the folder or user does not have an access to it.
+        if not folder:
             trans.response.status = 400
             return "Invalid folder id ( %s ) specified." % str( folder_id )
+        
+        if not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+            log.warning( "SECURITY: User (id: %s) without proper access rights is trying to load folder with ID of %s" % ( trans.user.id, folder.id ) )
+            trans.response.status = 400
+            return "Invalid folder id ( %s ) specified." % str( folder_id )
+        
+        path_to_root = []
+        def build_path ( folder ):
+            """
+            Search the path upwards recursively and load the whole route of names and ids for breadcrumb purposes.
+            """
+            path_to_root = []
+            # We are almost in root
+            if folder.parent_id is None:
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
+            else:
+            # We add the current folder and traverse up one folder.
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
+                upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_id )
+                path_to_root.extend( build_path( upper_folder ) )
+            return path_to_root
+            
+        # Return the reversed path so it starts with the library node.
+        full_path = build_path( folder )[::-1]
+        folder_container.append( dict( full_path = full_path ) )
+        
+        folder_contents = []
+        time_updated = ''
+        time_created = ''
+        # Go through every item in the folder and include its meta-data.
+        for content_item in self.load_folder_contents( trans, folder ):
+#             rval = content_item.to_dict()
+            return_item = {}
+            encoded_id = trans.security.encode_id( content_item.id )
+            time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+            time_created = content_item.create_time.strftime( "%Y-%m-%d %I:%M %p" )
+            
+            # For folder return also hierarchy values
+            if content_item.api_type == 'folder':
+                encoded_id = 'F' + encoded_id
+#                 time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+                return_item.update ( dict ( item_count = content_item.item_count ) )
 
-        for content in traverse( folder ):
-            encoded_id = trans.security.encode_id( content.id )
-            if content.api_type == 'folder':
-                encoded_id = 'F' + encoded_id
-            rval.append( dict( id = encoded_id,
-                               type = content.api_type,
-                               name = content.name,
-                               url = url_for( 'folder_contents', folder_id=encoded_id ) ) )
-        return rval
+            if content_item.api_type == 'file':
+                library_dataset_dict = content_item.to_dict()
+                library_dataset_dict['data_type']
+                library_dataset_dict['file_size']
+                library_dataset_dict['date_uploaded']
+                return_item.update ( dict ( data_type = library_dataset_dict['data_type'],
+                                            file_size = library_dataset_dict['file_size'],
+                                            date_uploaded = library_dataset_dict['date_uploaded'] ) )
+
+            # For every item return also the default meta-data
+            return_item.update( dict( id = encoded_id,
+                               type = content_item.api_type,
+                               name = content_item.name,
+                               time_updated = time_updated,
+                               time_created = time_created
+                                ) )
+            folder_contents.append( return_item )
+        # Put the data in the container
+        folder_container.append( dict( folder_contents = folder_contents ) )
+        return folder_container
 
     @web.expose_api
     def show( self, trans, id, library_id, **kwd ):
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- /dev/null
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -0,0 +1,241 @@
+"""
+API operations on the dataset from library.
+"""
+import glob
+import logging
+import operator
+import os
+import os.path
+import string
+import sys
+import tarfile
+import tempfile
+import urllib
+import urllib2
+import zipfile
+from galaxy.security import Action
+from galaxy import util, web
+from galaxy.util.streamball import StreamBall
+from galaxy.web.base.controller import BaseAPIController, UsesLibraryMixinItems
+
+import logging
+log = logging.getLogger( __name__ )
+
+# Test for available compression types
+# tmpd = tempfile.mkdtemp()
+# comptypes = []
+# for comptype in ( 'gz', 'bz2' ):
+#     tmpf = os.path.join( tmpd, 'compression_test.tar.' + comptype )
+#     try:
+#         archive = tarfile.open( tmpf, 'w:' + comptype )
+#         archive.close()
+#         comptypes.append( comptype )
+#     except tarfile.CompressionError:
+#         log.exception( "Compression error when testing %s compression.  This option will be disabled for library downloads." % comptype )
+#     try:
+#         os.unlink( tmpf )
+#     except OSError:
+#         pass
+ziptype = '32'
+# tmpf = os.path.join( tmpd, 'compression_test.zip' )
+# try:
+#     archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
+#     archive.close()
+#     comptypes.append( 'zip' )
+#     ziptype = '64'
+# except RuntimeError:
+#     log.exception( "Compression error when testing zip compression. This option will be disabled for library downloads." )
+# except (TypeError, zipfile.LargeZipFile):
+#     # ZIP64 is only in Python2.5+.  Remove TypeError when 2.4 support is dropped
+#     log.warning( 'Max zip file size is 2GB, ZIP64 not supported' )
+#     comptypes.append( 'zip' )
+# try:
+#     os.unlink( tmpf )
+# except OSError:
+#     pass
+# os.rmdir( tmpd )
+
+
+
+class DatasetsController( BaseAPIController, UsesLibraryMixinItems ):
+
+    @web.expose_api
+    def show( self, trans, id, **kwd ):
+        """
+        GET /api/libraries/datasets/{encoded_dataset_id}
+        Displays information about the dataset identified by the lda ID.
+        """
+        # Get dataset.
+        try:
+            dataset = self.get_library_dataset( trans, id = id )
+        except Exception, e:
+            return str( e )
+        try:
+            # Default: return dataset as dict.
+            rval = dataset.to_dict()
+        except Exception, e:
+            rval = "Error in dataset API at listing contents: " + str( e )
+            log.error( rval + ": %s" % str(e), exc_info=True )
+            trans.response.status = 500
+        
+        rval['id'] = trans.security.encode_id(rval['id']);
+        rval['ldda_id'] = trans.security.encode_id(rval['ldda_id']);
+        rval['folder_id'] = 'f' + trans.security.encode_id(rval['folder_id'])
+        
+        return rval
+    
+    @web.expose
+    def download( self, trans, format, **kwd ):
+        """
+        POST /api/libraries/datasets/download/{format}
+        POST data: ldda_ids = []
+        Downloads dataset(s) in the requested format.
+        """
+        lddas = []
+#         is_admin = trans.user_is_admin()
+#         current_user_roles = trans.get_current_user_roles()
+        
+        datasets_to_download = kwd['ldda_ids%5B%5D']
+        
+        if ( datasets_to_download != None ):
+            datasets_to_download = util.listify( datasets_to_download )
+            for dataset_id in datasets_to_download:
+                try:
+                    ldda = trans.sa_session.query( trans.app.model.LibraryDatasetDatasetAssociation ).get( trans.security.decode_id( dataset_id ) )
+                    assert not ldda.dataset.purged
+                    lddas.append( ldda )
+                except:
+                    ldda = None
+                    message += "Invalid library dataset id (%s) specified.  " % str( dataset_id )
+                    
+        if format in [ 'zip','tgz','tbz' ]:
+                error = False
+                killme = string.punctuation + string.whitespace
+                trantab = string.maketrans(killme,'_'*len(killme))
+                try:
+                    outext = 'zip'
+                    if format == 'zip':
+                        # Can't use mkstemp - the file must not exist first
+                        tmpd = tempfile.mkdtemp()
+                        util.umask_fix_perms( tmpd, trans.app.config.umask, 0777, self.app.config.gid )
+                        tmpf = os.path.join( tmpd, 'library_download.' + format )
+                        if ziptype == '64' and trans.app.config.upstream_gzip:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED, True )
+                        elif ziptype == '64':
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
+                        elif trans.app.config.upstream_gzip:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED )
+                        else:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED )
+                        archive.add = lambda x, y: archive.write( x, y.encode('CP437') )
+                    elif format == 'tgz':
+                        if trans.app.config.upstream_gzip:
+                            archive = StreamBall( 'w|' )
+                            outext = 'tar'
+                        else:
+                            archive = StreamBall( 'w|gz' )
+                            outext = 'tgz'
+                    elif format == 'tbz':
+                        archive = StreamBall( 'w|bz2' )
+                        outext = 'tbz2'
+                except ( OSError, zipfile.BadZipfile ):
+                    error = True
+                    log.exception( "Unable to create archive for download" )
+                    message = "Unable to create archive for download, please report this error"
+                    status = 'error'
+                except:
+                     error = True
+                     log.exception( "Unexpected error %s in create archive for download" % sys.exc_info()[0] )
+                     message = "Unable to create archive for download, please report - %s" % sys.exc_info()[0]
+                     status = 'error'
+                if not error:
+                    composite_extensions = trans.app.datatypes_registry.get_composite_extensions()
+                    seen = []
+                    for ldda in lddas:
+                        if ldda.dataset.state in [ 'new', 'upload', 'queued', 'running', 'empty', 'discarded' ]:
+                            continue
+                        ext = ldda.extension
+                        is_composite = ext in composite_extensions
+                        path = ""
+                        parent_folder = ldda.library_dataset.folder
+                        while parent_folder is not None:
+                            # Exclude the now-hidden "root folder"
+                            if parent_folder.parent is None:
+                                path = os.path.join( parent_folder.library_root[0].name, path )
+                                break
+                            path = os.path.join( parent_folder.name, path )
+                            parent_folder = parent_folder.parent
+                        path += ldda.name
+                        while path in seen:
+                            path += '_'
+                        seen.append( path )
+                        zpath = os.path.split(path)[-1] # comes as base_name/fname
+                        outfname,zpathext = os.path.splitext(zpath)
+                        if is_composite:
+                            # need to add all the components from the extra_files_path to the zip
+                            if zpathext == '':
+                                zpath = '%s.html' % zpath # fake the real nature of the html file
+                            try:
+                                archive.add(ldda.dataset.file_name,zpath) # add the primary of a composite set
+                            except IOError:
+                                error = True
+                                log.exception( "Unable to add composite parent %s to temporary library download archive" % ldda.dataset.file_name)
+                                message = "Unable to create archive for download, please report this error"
+                                status = 'error'
+                                continue
+                            flist = glob.glob(os.path.join(ldda.dataset.extra_files_path,'*.*')) # glob returns full paths
+                            for fpath in flist:
+                                efp,fname = os.path.split(fpath)
+                                if fname > '':
+                                    fname = fname.translate(trantab)
+                                try:
+                                    archive.add( fpath,fname )
+                                except IOError:
+                                    error = True
+                                    log.exception( "Unable to add %s to temporary library download archive %s" % (fname,outfname))
+                                    message = "Unable to create archive for download, please report this error"
+                                    status = 'error'
+                                    continue
+                        else: # simple case
+                            try:
+                                archive.add( ldda.dataset.file_name, path )
+                            except IOError:
+                                error = True
+                                log.exception( "Unable to write %s to temporary library download archive" % ldda.dataset.file_name)
+                                message = "Unable to create archive for download, please report this error"
+                                status = 'error'
+                    if not error:
+                        lname = 'selected_dataset'
+                        fname = lname.replace( ' ', '_' ) + '_files'
+                        if format == 'zip':
+                            archive.close()
+                            trans.response.set_content_type( "application/octet-stream" )
+                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                            archive = util.streamball.ZipBall(tmpf, tmpd)
+                            archive.wsgi_status = trans.response.wsgi_status()
+                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            return archive.stream
+                        else:
+                            trans.response.set_content_type( "application/x-tar" )
+                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                            archive.wsgi_status = trans.response.wsgi_status()
+                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            return archive.stream
+        elif format == 'uncompressed':
+            if len(lddas) != 1:
+                return 'Wrong request'
+            else:
+                single_dataset = lddas[0]
+                trans.response.set_content_type( single_dataset.get_mime() )
+                fStat = os.stat( ldda.file_name )
+                trans.response.headers[ 'Content-Length' ] = int( fStat.st_size )
+                valid_chars = '.,^_-()[]0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
+                fname = ldda.name
+                fname = ''.join( c in valid_chars and c or '_' for c in fname )[ 0:150 ]
+                trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s"' % fname
+                try:
+                    return open( single_dataset.file_name )
+                except:
+                    return 'This dataset contains no content'
+        else:
+            return 'Wrong format';
\ No newline at end of file
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 lib/galaxy/webapps/galaxy/api/libraries.py
--- a/lib/galaxy/webapps/galaxy/api/libraries.py
+++ b/lib/galaxy/webapps/galaxy/api/libraries.py
@@ -49,9 +49,10 @@
                            trans.model.Library.table.c.id.in_( accessible_restricted_library_ids ) ) )
         rval = []
         for library in query:
-            item = library.to_dict()
+            item = library.to_dict( view='element' )
             item['url'] = url_for( route, id=trans.security.encode_id( library.id ) )
-            item['id'] = trans.security.encode_id( item['id'] )
+            item['id'] = 'F' + trans.security.encode_id( item['id'] )
+            item['root_folder_id'] = 'F' + trans.security.encode_id( item['root_folder_id'] )
             rval.append( item )
         return rval
 
@@ -131,6 +132,9 @@
         rval['name'] = name
         rval['id'] = encoded_id
         return rval
+    
+    def edit( self, trans, payload, **kwd  ):
+        return "Not implemented yet"
 
     @web.expose_api
     def delete( self, trans, id, **kwd ):
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -46,12 +46,6 @@
     atexit.register( app.shutdown )
     # Create the universe WSGI application
     webapp = GalaxyWebApplication( app, session_cookie='galaxysession', name='galaxy' )
-    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
-    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
-                      controller='admin_toolshed',
-                      action='display_image_in_repository',
-                      repository_id=None,
-                      image_file=None )
     webapp.add_ui_controllers( 'galaxy.webapps.galaxy.controllers', app )
     # Force /history to go to /root/history -- needed since the tests assume this
     webapp.add_route( '/history', controller='root', action='history' )
@@ -75,22 +69,12 @@
     webapp.add_route( '/u/:username/v/:slug', controller='visualization', action='display_by_username_and_slug' )
     webapp.add_route( '/search', controller='search', action='index' )
 
-    # Add the web API
+    # ================
+    # =====  API =====
+    # ================
+
     webapp.add_api_controllers( 'galaxy.webapps.galaxy.api', app )
-    # The /folders section is experimental at this point:
-    log.debug( "app.config.api_folders: %s" % app.config.api_folders )
-    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
-    webapp.mapper.resource( 'content', 'contents',
-                                controller='folder_contents',
-                                name_prefix='folder_',
-                                path_prefix='/api/folders/:folder_id',
-                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
-    webapp.mapper.resource( 'content',
-                                'contents',
-                                controller='library_contents',
-                                name_prefix='library_',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+
     webapp.mapper.resource( 'content',
                                 'contents',
                                 controller='history_contents',
@@ -102,10 +86,6 @@
                               controller="datasets",
                               action="display",
                               conditions=dict(method=["GET"]))
-    webapp.mapper.resource( 'permission',
-                                'permissions',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
     webapp.mapper.resource( 'user',
                                 'users',
                                 controller='group_users',
@@ -127,11 +107,6 @@
     _add_item_tags_controller( webapp,
                                name_prefix="workflow_",
                                path_prefix='/api/workflows/:workflow_id' )
-
-    _add_item_extended_metadata_controller( webapp,
-                               name_prefix="library_dataset_",
-                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
-
     _add_item_annotation_controller( webapp,
                                name_prefix="history_content_",
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
@@ -141,7 +116,6 @@
     _add_item_annotation_controller( webapp,
                                name_prefix="workflow_",
                                path_prefix='/api/workflows/:workflow_id' )
-
     _add_item_provenance_controller( webapp,
                                name_prefix="history_content_",
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
@@ -171,6 +145,39 @@
     #webapp.mapper.connect( 'run_workflow', '/api/workflow/{workflow_id}/library/{library_id}', controller='workflows', action='run', workflow_id=None, library_id=None, conditions=dict(method=["GET"]) )
     webapp.mapper.resource( 'search', 'search', path_prefix='/api' )
 
+    # =======================
+    # ===== LIBRARY API =====
+    # =======================
+    
+    # The /folders section is experimental at this point:
+    log.debug( "app.config.api_folders: %s" % app.config.api_folders )
+
+    webapp.mapper.connect( 'show_lda_item', '/api/libraries/datasets/:id', controller='lda_datasets', action='show', conditions=dict(method=["GET"]) )
+    webapp.mapper.connect( 'download_lda_items', '/api/libraries/datasets/download/:format', controller='lda_datasets', action='download', conditions=dict(method=["POST"]) )
+
+    webapp.mapper.resource_with_deleted( 'library', 'libraries', path_prefix='/api' )
+    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
+
+    webapp.mapper.resource( 'content', 'contents',
+                                controller='folder_contents',
+                                name_prefix='folder_',
+                                path_prefix='/api/folders/:folder_id',
+                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
+    webapp.mapper.resource( 'content',
+                                'contents',
+                                controller='library_contents',
+                                name_prefix='library_',
+                                path_prefix='/api/libraries/:library_id',
+                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    webapp.mapper.resource( 'permission',
+                                'permissions',
+                                path_prefix='/api/libraries/:library_id',
+                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    _add_item_extended_metadata_controller( webapp,
+                               name_prefix="library_dataset_",
+                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
+    
+
     # add as a non-ATOM API call to support the notion of a 'current/working' history unique to the history resource
     webapp.mapper.connect( "set_as_current", "/api/histories/{id}/set_as_current",
         controller="histories", action="set_as_current", conditions=dict( method=["POST"] ) )
@@ -188,6 +195,18 @@
     webapp.mapper.connect("workflow_dict", '/api/workflows/{workflow_id}/download', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
     # Preserve the following download route for now for dependent applications  -- deprecate at some point
     webapp.mapper.connect("workflow_dict", '/api/workflows/download/{workflow_id}', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
+    
+    # ====================
+    # ===== TOOLSHED =====
+    # ====================
+    
+    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
+    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
+                      controller='admin_toolshed',
+                      action='display_image_in_repository',
+                      repository_id=None,
+                      image_file=None )
+    
     # Galaxy API for tool shed features.
     webapp.mapper.resource( 'tool_shed_repository',
                             'tool_shed_repositories',
@@ -201,6 +220,7 @@
                             path_prefix='/api',
                             new={ 'install_repository_revision' : 'POST' },
                             parent_resources=dict( member_name='tool_shed_repository', collection_name='tool_shed_repositories' ) )
+
     # Connect logger from app
     if app.trace_logger:
         webapp.trace_logger = app.trace_logger
@@ -221,7 +241,7 @@
         galaxy.model.mapping.metadata.engine.connection_provider._pool.dispose()
     except:
         pass
-    # Close any pooled database connections before forking
+     # Close any pooled database connections before forking
     try:
         galaxy.model.tool_shed_install.mapping.metadata.engine.connection_provider._pool.dispose()
     except:
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 lib/galaxy/webapps/galaxy/controllers/library.py
--- a/lib/galaxy/webapps/galaxy/controllers/library.py
+++ b/lib/galaxy/webapps/galaxy/controllers/library.py
@@ -76,6 +76,17 @@
 
     library_list_grid = LibraryListGrid()
 
+    
+    @web.expose
+    def list( self, trans, **kwd ):
+        params = util.Params( kwd )
+        # define app configuration for generic mako template
+        app = {
+            'jscript'       : "galaxy.library"
+        }
+        # fill template
+        return trans.fill_template('galaxy.panels.mako', config = {'app' : app})
+
     @web.expose
     def index( self, trans, **kwd ):
         params = util.Params( kwd )
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 static/scripts/galaxy.library.js
--- /dev/null
+++ b/static/scripts/galaxy.library.js
@@ -0,0 +1,860 @@
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+// === GALAXY LIBRARY MODULE ====
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+
+// global variables
+var view            = null;
+var library_router  = null;
+var responses = [];
+
+
+// load required libraries
+// require([
+//     // load js libraries
+//     'utils/galaxy.css',
+//     ], function(css){
+//         // load css
+//         css.load_file("static/style/library.css");
+// });
+
+// dependencies
+define(["galaxy.modal", "galaxy.masthead", "utils/galaxy.utils"], function(mod_modal, mod_masthead, mod_utils) {
+
+// MMMMMMMMMMMMMMM
+// === Models ====
+// MMMMMMMMMMMMMMM
+
+    // LIBRARY
+    var Library = Backbone.Model.extend({
+      urlRoot: '/api/libraries'
+  });
+
+    // LIBRARIES
+    var Libraries = Backbone.Collection.extend({
+      url: '/api/libraries',
+      model: Library
+  });
+
+    // ITEM
+    var Item = Backbone.Model.extend({
+        urlRoot : '/api/libraries/datasets'
+    })
+
+    // FOLDER
+    var Folder = Backbone.Collection.extend({
+      model: Item      
+  })
+
+    // CONTAINER for folder contents (folders, items and metadata).
+    var FolderContainer = Backbone.Model.extend({
+     defaults : {
+        folder : new Folder(),
+        full_path : "unknown",
+        urlRoot : "/api/folders/",
+        id : "unknown"
+    },
+    parse : function(obj) {
+      this.full_path = obj[0].full_path;
+          // update the inner collection
+          this.get("folder").reset(obj[1].folder_contents);
+          return obj;
+      }
+  })
+
+    // HISTORY ITEM
+    var HistoryItem = Backbone.Model.extend({
+        urlRoot : '/api/histories/'
+    });
+
+    // HISTORY
+    var GalaxyHistory = Backbone.Model.extend({
+        url : '/api/histories/'
+    });
+
+    // HISTORIES
+    var GalaxyHistories = Backbone.Collection.extend({
+        url : '/api/histories',
+        model : GalaxyHistory
+    });
+
+    //ROUTER
+    var LibraryRouter = Backbone.Router.extend({    
+        routes: {
+            "" :                 "libraries",
+            "folders/:id" :      "folder_content",
+            "folders/:folder_id/download/:format" : "download"
+        }
+    });
+
+
+// MMMMMMMMMMMMMM
+// === VIEWS ====
+// MMMMMMMMMMMMMM
+
+// galaxy folder
+var FolderContentView = Backbone.View.extend({
+    // main element definition
+    el : '#center',
+    // progress percentage
+    progress: 0,
+    // progress rate per one item
+    progressStep: 1, 
+    // last selected history in modal for UX
+    lastSelectedHistory: '',
+    // self modal
+    modal : null,
+    // loaded folders
+    folders : null,
+    
+    // initialize
+    initialize : function(){
+        this.folders = [];
+        this.queue = jQuery.Deferred();
+        this.queue.resolve();
+    },    
+
+// MMMMMMMMMMMMMMMMMM
+// === TEMPLATES ====
+// MMMMMMMMMMMMMMMMMM
+
+    // set up 
+    templateFolder : function (){
+        var tmpl_array = [];
+
+        // CONTAINER
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; ">');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
+
+        // TOOLBAR
+        tmpl_array.push('<div id="library_folder_toolbar" >');
+        tmpl_array.push('   <button title="Create New Folder" id="toolbtn_create_folder" class="btn btn-primary" type="button"><span class="fa fa-icon-plus"></span><span class="fa fa-icon-folder-close"></span> folder</button>');
+        tmpl_array.push('   <button id="toolbtn_bulk_import" class="btn btn-primary" style="display: none; margin-left: 0.5em;" type="button"><span class="fa fa-icon-external-link"></span> to history</button>');
+        
+        tmpl_array.push('   <div id="toolbtn_dl" class="btn-group" style="margin-left: 0.5em; display: none; ">');
+        tmpl_array.push('       <button id="drop_toggle" type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown">');
+        tmpl_array.push('       <span class="fa fa-icon-download"></span> download <span class="caret"></span>');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       <ul class="dropdown-menu" role="menu">');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tgz">.tar.gz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tbz">.tar.bz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/zip">.zip</a></li>');
+        tmpl_array.push('       </ul>');
+        tmpl_array.push('   </div>');
+
+        tmpl_array.push('</div>');
+
+        // BREADCRUMBS
+        tmpl_array.push('<div class="library_breadcrumb">');
+        tmpl_array.push('<a title="Return to the list of libraries" href="#">Libraries</a><b>|</b> ');
+        tmpl_array.push('<% _.each(path, function(path_item) { %>');
+        tmpl_array.push('<% if (path_item[0] != id) { %>');
+        tmpl_array.push('<a title="Return to this folder" href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a><b>|</b> ');
+        tmpl_array.push('<% } else { %>');
+        tmpl_array.push('<span title="You are in this folder"><%- path_item[1] %></span>');
+        tmpl_array.push('<% } %>');
+        tmpl_array.push('<% }); %>');
+        tmpl_array.push('</div>');
+
+        // FOLDER CONTENT
+        tmpl_array.push('<table id="folder_table" class="table table-condensed">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
+        tmpl_array.push('       <th class="button_heading">view</th>');
+        tmpl_array.push('       <th>name</th>');
+        tmpl_array.push('       <th>data type</th>');
+        tmpl_array.push('       <th>size</th>');
+        tmpl_array.push('       <th>date</th>');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td><button title="Go to parent folder" type="button" data-id="<%- upper_folder_id %>" class="btn_open_folder btn btn-default btn-xs">');        
+        tmpl_array.push('       <span class="fa fa-icon-arrow-up"></span> .. go up</td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       <td></td>');        
+        tmpl_array.push('       </tr>');        
+        tmpl_array.push('       <% _.each(items, function(content_item) { %>');
+        tmpl_array.push('       <tr class="folder_row light" id="<%- content_item.id %>">');
+        tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>'); // folder
+        tmpl_array.push('               <td></td>');
+        tmpl_array.push('               <td><button title="Open this folder" type="button" data-id="<%- content_item.id %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- content_item.get("name") %>');
+        tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>'); // empty folder
+        tmpl_array.push('           <span class="muted">(empty folder)</span>');
+        tmpl_array.push('           <% } %>');
+        tmpl_array.push('           </td>');
+        tmpl_array.push('           <td>folder</td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("item_count")) %> item(s)</td>'); // size
+        tmpl_array.push('           <% } else {  %>');
+        tmpl_array.push('           <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
+        tmpl_array.push('       <td>');
+        tmpl_array.push('       <button title="See details of this dataset" type="button" class="library-dataset btn btn-default btn-xs">');
+        tmpl_array.push('       <span class="fa fa-icon-eye-open"></span> details');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       </td>');
+        tmpl_array.push('           <td><%- content_item.get("name") %></td>'); // dataset
+        tmpl_array.push('           <td><%= _.escape(content_item.get("data_type")) %></td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("readable_size")) %></td>'); // size
+        tmpl_array.push('           <% } %>  ');
+        tmpl_array.push('           <td><%= _.escape(content_item.get("time_updated")) %></td>'); // time updated
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('       ');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+
+        tmpl_array.push('</div>');
+        return tmpl_array.join('');
+    },
+    templateDatasetModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<div id="dataset_info_modal">');
+        tmpl_array.push('   <table class="table table-striped table-condensed">');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row" id="id_row" data-id="<%= _.escape(item.get("ldda_id")) %>">Name</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("name")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Data type</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("data_type")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Genome build</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("genome_build")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('           <th scope="row">Size</th>');
+        tmpl_array.push('           <td><%= _.escape(size) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Date uploaded</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("date_uploaded")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Uploaded by</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("uploaded_by")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('           <tr scope="row">');
+        tmpl_array.push('           <th scope="row">Data Lines</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_data_lines")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <th scope="row">Comment Lines</th>');
+        tmpl_array.push('           <% if (item.get("metadata_comment_lines") === "") { %>'); //folder        
+        tmpl_array.push('               <td scope="row"><%= _.escape(item.get("metadata_comment_lines")) %></td>');
+        tmpl_array.push('           <% } else { %>'); 
+        tmpl_array.push('               <td scope="row">unknown</td>');    
+        tmpl_array.push('           <% } %>'); 
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Number of Columns</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_columns")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Column Types</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_column_types")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Miscellaneous information</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("misc_blurb")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('   </table>');
+        tmpl_array.push('   <pre class="peek">');
+        tmpl_array.push('   </pre>');
+        tmpl_array.push('</div>');
+
+        return tmpl_array.join('');
+    },
+
+    templateHistorySelectInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_combo" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_single" name="dataset_import_single" style="width:50%; margin-bottom: 1em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
+
+        return tmpl_array.join('');
+        },   
+
+    templateBulkImportInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_combo_bulk" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_bulk" name="dataset_import_bulk" style="width:50%; margin-bottom: 1em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
+
+        return tmpl_array.join('');
+        },    
+
+      // convert size to nice string
+      size_to_string : function (size)
+      {
+            // identify unit
+            var unit = "";
+            if (size >= 100000000000)   { size = size / 100000000000; unit = "TB"; } else
+            if (size >= 100000000)      { size = size / 100000000; unit = "GB"; } else
+            if (size >= 100000)         { size = size / 100000; unit = "MB"; } else
+            if (size >= 100)            { size = size / 100; unit = "KB"; } else
+            { size = size * 10; unit = "b"; }
+            // return formatted string
+            return (Math.round(size) / 10) + unit;
+        },
+
+// MMMMMMMMMMMMMMM
+// === EVENTS ====
+// MMMMMMMMMMMMMMM
+
+      // event binding
+      events: {
+            'click #select-all-checkboxes' : 'selectAll',
+            'click .folder_row' : 'selectClicked',
+            'click #toolbtn_bulk_import' : 'modalBulkImport',
+            'click #toolbtn_dl' : 'bulkDownload',
+            'click .library-dataset' : 'showDatasetDetails',
+            'click #toolbtn_create_folder' : 'createFolderModal',
+            'click .btn_open_folder' : 'navigateToFolder'
+        },
+
+      //render the folder view
+      render: function (options) {
+        //hack to show scrollbars
+        $("#center").css('overflow','auto');
+
+        view = this;
+        var that = this;
+
+        var folderContainer = new FolderContainer({id: options.id});
+        folderContainer.url = folderContainer.attributes.urlRoot + options.id + '/contents';
+
+        folderContainer.fetch({
+          success: function (container) {
+
+            // prepare nice size strings
+              for (var i = 0; i < folderContainer.attributes.folder.models.length; i++) {
+                  var model = folderContainer.attributes.folder.models[i]
+                  if (model.get('type') === 'file'){
+                    model.set('readable_size', that.size_to_string(model.get('file_size')))
+                  }
+              };
+
+              // find the upper id
+              var path = folderContainer.full_path;
+              var upper_folder_id;
+              if (path.length === 1){ // library is above us
+                upper_folder_id = 0;
+              } else {
+                upper_folder_id = path[path.length-2][0];
+              }
+
+              var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id, upper_folder_id: upper_folder_id });
+              // var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id });
+              that.$el.html(template);
+            }
+          })
+        },
+
+      // handles the click on 'open' and 'upper' folder icons
+      navigateToFolder : function(event){
+        var folder_id = $(event.target).attr('data-id');
+        if (typeof folder_id === 'undefined') {
+            return false;
+        } else if (folder_id === '0'){
+            library_router.navigate('#', {trigger: true, replace: true});
+        } else {
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+        }
+      },
+
+      //show modal with current dataset info
+      showDatasetDetails : function(event){
+        // prevent default
+        event.preventDefault();
+        
+//TODO check whether we already have the data
+
+        //load the ID of the row
+        var id = $(event.target).parent().parent().attr('id');
+        
+        //create new item
+        var item = new Item();
+        var histories = new GalaxyHistories();
+        item.id = id;
+        var self = this;
+
+        //fetch the dataset info
+        item.fetch({
+          success: function (item) {
+// TODO can render here already
+                //fetch user histories for import purposes
+                histories.fetch({
+                    success: function (histories){self.renderModalAfterFetch(item, histories)}
+                });
+            }
+        });
+    },
+
+      // show the current dataset in a modal
+      renderModalAfterFetch : function(item, histories){
+        var size = this.size_to_string(item.get('file_size'));
+        var template = _.template(this.templateDatasetModal(), { item : item, size : size });
+        this.modal = null;
+            // make modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal({
+                title   : 'Dataset Details',
+                body    : template,
+                buttons : {
+                    'Import' : function() { self.importCurrentIntoHistory() },
+                    'Download' : function() { self.downloadCurrent() },
+                    'Close'  : function() { self.modal.hide(); $('.modal').remove(); self.modal = null; } // TODO refill nicely modal with data
+                }
+            });
+            $(".peek").html(item.get("peek"));
+            var history_footer_tmpl = _.template(this.templateHistorySelectInModal(), {histories : histories.models});
+            $(this.modal.elMain).find('.buttons').prepend(history_footer_tmpl);
+
+            // preset last selected history if we know it
+            if (self.lastSelectedHistory.length > 0) {
+                $(this.modal.elMain).find('#dataset_import_single').val(self.lastSelectedHistory);
+            }
+
+            // show the prepared modal
+            this.modal.show();
+        },
+
+        // download dataset shown currently in modal
+        downloadCurrent : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
+
+            var library_dataset_id = [];
+            library_dataset_id.push($('#id_row').attr('data-id'));
+            var url = '/api/libraries/datasets/download/uncompressed';
+            var data = {'ldda_ids' : library_dataset_id};
+
+            // we assume the view is existent
+            folderContentView.processDownload(url, data);
+            this.modal.enableButton('Import');
+            this.modal.enableButton('Download');
+        },
+
+        // import dataset shown currently in modal into selected history
+        importCurrentIntoHistory : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
+
+            var history_id = $(this.modal.elMain).find('select[name=dataset_import_single] option:selected').val();
+            this.lastSelectedHistory = history_id; //save selected history for further use
+
+            var library_dataset_id = $('#id_row').attr('data-id');
+            var historyItem = new HistoryItem();
+            var self = this;
+            historyItem.url = historyItem.urlRoot + history_id + '/contents';
+
+            // save the dataset into selected history
+            historyItem.save({ content : library_dataset_id, source : 'library' }, { success : function(){
+                self.modal.showNotification('Dataset imported', 3000, 'success');
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');
+            }, error : function(){
+                self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, 'error');
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');                
+            }
+        });
+        },
+
+        // select all datasets 
+        selectAll : function (event) {
+           var selected = event.target.checked;
+                 // Iterate each checkbox
+                 $(':checkbox').each(function () { this.checked = selected; });
+                 this.showTools();
+             },
+
+         // click checkbox on folder click
+         selectClicked : function (event) {
+            var checkbox = $("#" + event.target.parentElement.id).find(':checkbox')
+                if (checkbox[0] != undefined) {
+                  if (checkbox[0].checked){
+                    checkbox[0].checked = '';
+                    // $(event.target.parentElement).css('background-color', '').css('color', '');
+                    $(event.target.parentElement).removeClass('dark');
+                    $(event.target.parentElement).find('a').removeClass('dark');
+                    $(event.target.parentElement).addClass('light');
+                    $(event.target.parentElement).find('a').addClass('light');
+                } else {
+                    checkbox[0].checked = 'selected';
+                    $(event.target.parentElement).removeClass('light');
+                    $(event.target.parentElement).find('a').removeClass('light');
+                    $(event.target.parentElement).addClass('dark');
+                    $(event.target.parentElement).find('a').addClass('dark');
+                    // $(event.target.parentElement).css('background-color', '#8389a1').css('color', 'white');
+                }
+            }
+            this.showTools();
+        },
+
+        // show toolbar in case something is selected
+        showTools : function(){
+            var checkedValues = $('#folder_table').find(':checked');
+            if(checkedValues.length > 0){
+                $('#toolbtn_bulk_import').show();
+                $('#toolbtn_dl').show();
+            } else {
+                $('#toolbtn_bulk_import').hide();
+                $('#toolbtn_dl').hide();
+            }
+
+        },
+
+        // show bulk import modal
+        modalBulkImport : function(){
+            var self = this;
+            // fetch histories
+            var histories = new GalaxyHistories();
+            histories.fetch({
+                    success: function (histories){
+                        // make modal
+                        var history_modal_tmpl =  _.template(self.templateBulkImportInModal(), {histories : histories.models});
+                        self.modal = new mod_modal.GalaxyModal({
+                            title   : 'Import into History',
+                            body    : history_modal_tmpl,
+                            buttons : {
+                                'Import' : function() {self.importAllIntoHistory()},
+                                'Close'  : function() {self.modal.hide(); $('.modal').remove(); self.modal = null;}
+                            }
+                        });
+                        // show the prepared modal
+                        self.modal.show();
+                    }
+                });
+        },
+
+        // import all selected datasets into history
+        importAllIntoHistory : function (){
+            //disable the button
+            this.modal.disableButton('Import');
+
+            var history_id = $("select[name=dataset_import_bulk] option:selected").val();
+            var history_name = $("select[name=dataset_import_bulk] option:selected").text();
+
+            var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+            var progress_bar_tmpl = _.template(this.templateProgressBar(), { history_name : history_name });
+            $(this.modal.elMain).find('.modal-body').html(progress_bar_tmpl);
+            
+            // init the progress bar
+            var progressStep = 100 / dataset_ids.length;
+            this.initProgress(progressStep);
+
+            // prepare the dataset objects to be imported
+            var datasets_to_import = [];
+            for (var i = dataset_ids.length - 1; i >= 0; i--) {
+                library_dataset_id = dataset_ids[i];
+                var historyItem = new HistoryItem();
+                var self = this;
+                historyItem.url = historyItem.urlRoot + history_id + '/contents';
+                historyItem.content = library_dataset_id;
+                historyItem.source = 'library';
+                datasets_to_import.push(historyItem);
+            };
+
+            // call the recursive function to call ajax one after each other
+            this.chainCall(datasets_to_import);
+        },
+
+        chainCall: function(history_item_set){
+            var self = this;
+            var popped_item = history_item_set.pop();
+            if (typeof popped_item === "undefined") {
+                self.modal.showNotification('All datasets imported', 3000, 'success');
+                // enable button again
+                self.modal.enableButton('Import');
+                return
+            }
+                var promise = $.when(popped_item.save({content: popped_item.content, source: popped_item.source})).done(function(a1){
+                        self.updateProgress();
+                        responses.push(a1);
+                        self.chainCall(history_item_set);
+                    });
+        },
+
+        initProgress: function(progressStep){
+            this.progress = 0;
+            this.progressStep = progressStep;
+        },
+        updateProgress: function(){
+            this.progress += this.progressStep;
+            $('.progress-bar').width(Math.round(this.progress) + '%');
+            txt_representation = Math.round(this.progress) + '% Complete';
+            $('.completion_span').text(txt_representation);
+        },
+
+      // progress bar 
+      templateProgressBar : function (){
+        var tmpl_array = [];
+
+        tmpl_array.push('<div class="import_text">');
+        tmpl_array.push('Importing selected datasets to history <b><%= _.escape(history_name) %></b>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('<div class="progress">');
+        tmpl_array.push('   <div class="progress-bar" role="progressbar" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100" style="width: 00%;">');
+        tmpl_array.push('       <span class="completion_span">0% Complete</span>');
+        tmpl_array.push('   </div>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('');
+
+        return tmpl_array.join('');
+      },
+
+      // download selected datasets
+      download : function(folder_id, format){
+        var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+
+        var url = '/api/libraries/datasets/download/' + format;
+        var data = {'ldda_ids' : dataset_ids};
+        this.processDownload(url, data);
+      },
+
+      // create hidden form and submit through POST to initialize download
+      processDownload: function(url, data, method){
+        //url and data options required
+        if( url && data ){ 
+                //data can be string of parameters or array/object
+                data = typeof data == 'string' ? data : $.param(data);
+                //split params into form inputs
+                var inputs = '';
+                $.each(data.split('&'), function(){ 
+                        var pair = this.split('=');
+                        inputs+='<input type="hidden" name="'+ pair[0] +'" value="'+ pair[1] +'" />'; 
+                });
+                //send request
+                $('<form action="'+ url +'" method="'+ (method||'post') +'">'+inputs+'</form>')
+                .appendTo('body').submit().remove();
+        };
+      },
+
+      // shows modal for creating folder
+      createFolderModal: function(){
+        alert('creating folder');
+      }
+
+    });
+
+// galaxy library view
+var GalaxyLibraryview = Backbone.View.extend({
+    el: '#center',
+
+    events: {
+        'click #create_new_library_btn' : 'show_library_modal'
+    },
+
+    // initialize
+    initialize : function(){
+    },    
+
+    // template
+    template_library_list : function (){
+        tmpl_array = [];
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; overflow: auto !important; ">');
+
+        tmpl_array.push('');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
+        tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary icon-file ">New Library</a>');
+        tmpl_array.push('<table class="table table-condensed">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('     <th class="button_heading"></th>');
+        tmpl_array.push('     <th>name</th>');
+        tmpl_array.push('     <th>description</th>');
+        tmpl_array.push('     <th>synopsis</th> ');
+        tmpl_array.push('     <th>model type</th> ');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <% _.each(libraries, function(library) { %>');
+        tmpl_array.push('           <tr>');
+        tmpl_array.push('               <td><button title="Open this library" type="button" data-id="<%- library.get("root_folder_id") %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- library.get("name") %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
+        tmpl_array.push('           </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+        
+        tmpl_array.push('</div>');
+        return tmpl_array.join('');
+    },
+
+    // render
+    render: function () {
+        //hack to show scrollbars
+        $("#center").css('overflow','auto');
+        var that = this;
+        // if (typeof libraries === "undefined") {
+          libraries = new Libraries();
+        // }
+        libraries.fetch({
+          success: function (libraries) {
+            var template = _.template(that.template_library_list(), { libraries : libraries.models });
+            that.$el.html(template);
+        }
+    })
+    },
+
+    // own modal
+    modal : null,
+
+    // show/hide create library modal
+    show_library_modal : function (event){
+        event.preventDefault();
+        event.stopPropagation();
+        
+        // create modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal(
+            {
+                title   : 'Create New Library',
+                body    : this.template_new_library(),
+                buttons : {
+                    'Create' : function() {self.create_new_library_event()},
+                    'Close'  : function() {self.modal.hide()}
+                }
+            });
+        
+        // show prepared modal
+        this.modal.show();
+    },
+
+    // create the new library from modal
+    create_new_library_event: function(){
+        var libraryDetails = this.serialize_new_library();
+        var valid = this.validate_new_library(libraryDetails);
+        var library = new Library();
+        var self = this;
+        library.save(libraryDetails, {
+          success: function (library) {
+            self.modal.hide();
+            self.clear_library_modal();
+            self.render();
+          },
+          error: function(){
+            self.modal.showNotification('An error occured', 5000, 'error');
+            }
+    });
+        return false;
+    },
+
+    // clear the library modal once saved
+    clear_library_modal : function(){
+        $("input[name='Name']").val('');
+        $("input[name='Description']").val('');
+        $("input[name='Synopsis']").val('');
+    },
+
+    // serialize data from the form
+    serialize_new_library : function(){
+        return {
+            name: $("input[name='Name']").val(),
+            description: $("input[name='Description']").val(),
+            synopsis: $("input[name='Synopsis']").val()
+        };
+    },
+
+    validate_new_library: function(library){
+
+    },
+
+
+    // template for new library modal
+    template_new_library: function()
+    {
+        tmpl_array = [];
+
+        tmpl_array.push('<div id="new_library_modal">');
+        tmpl_array.push('<form>');
+        tmpl_array.push('<input type="text" name="Name" value="" placeholder="Name">');
+        tmpl_array.push('<input type="text" name="Description" value="" placeholder="Description">');
+        tmpl_array.push('<input type="text" name="Synopsis" value="" placeholder="Synopsis">');
+        tmpl_array.push('</form>');
+        tmpl_array.push('</div>');
+
+        return tmpl_array.join('');
+    }
+});
+
+// galaxy library wrapper View
+var GalaxyLibrary = Backbone.View.extend({
+    folderContentView : null,
+    galaxyLibraryview : null,
+    initialize : function(){
+
+        folderContentView = new FolderContentView();
+        galaxyLibraryview = new GalaxyLibraryview();
+
+        library_router = new LibraryRouter();
+
+        library_router.on('route:libraries', function() {
+          // render libraries list
+          galaxyLibraryview.render();
+      });
+
+        library_router.on('route:folder_content', function(id) {
+          // render folder's contents
+          folderContentView.render({id: id});
+      });    
+
+       library_router.on('route:download', function(folder_id, format) {
+          // send download stream
+          if (typeof folderContentView === 'undefined'){
+            alert('you cant touch this!');
+          // } else if (folderContentView.modal !== null){
+            // folderContentView.download(folder_id, format);
+          } else if ($('#center').find(':checked').length === 0) { // coming from outside of the library app
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+            // TODO show message of redirection
+          } else {
+            folderContentView.download(folder_id, format);
+            library_router.navigate('folders/' + folder_id, {trigger: false, replace: true});
+          }
+
+      });      
+
+Backbone.history.start();   
+
+return this
+}
+});
+
+// return
+return {
+    GalaxyApp: GalaxyLibrary
+};
+
+});
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 static/scripts/galaxy.menu.js
--- a/static/scripts/galaxy.menu.js
+++ b/static/scripts/galaxy.menu.js
@@ -57,7 +57,12 @@
 
         tab_shared.add({
             title   : "Data Libraries",
-            content : "library/index",
+            content : "library/index"
+        });
+
+        tab_shared.add({
+            title   : "New Libraries",
+            content : "library/list",
             divider : true
         });
 
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -1,6 +1,3 @@
-/*
-    galaxy modal
-*/
 
 // dependencies
 define([], function() {
@@ -25,8 +22,51 @@
     
     // initialize
     initialize : function(options) {
+        self = this;
         if (options)
             this.create(options);
+
+        this.bindClick(event, self);
+    },
+
+    // bind the click-to-hide function
+    bindClick: function(event, that) {
+        // bind the ESC key to hide() function
+        $(document).on('keyup', function(event){
+            if (event.keyCode == 27) { self.hide(); }
+        })
+        // bind the click anywhere to hide() function...
+        $('html').on('click', function(event){
+            that.hide();
+        })
+        // ...but don't hide if the click is on modal content
+        $('.modal-content').on('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+    // unbind the click-to-hide function
+    unbindClick: function(event, that){
+        // bind the ESC key to hide() function
+        $(document).off('keyup', function(event){
+            if (event.keyCode == 27) { that.hide(); }
+        })
+        // unbind the click anywhere to hide() function...
+        $('html').off('click', function(event){
+            that.hide();
+        })
+        // ...but don't hide if the click is on modal content
+        $('.modal-content').off('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+
+    // destroy
+    destroy : function(){
+        this.hide();
+        this.unbindClick();
+        $('.modal').remove();
     },
 
     // adds and displays a new frame/window
@@ -87,6 +127,7 @@
         this.$footer  = (this.$el).find('.modal-footer');
         this.$buttons = (this.$el).find('.buttons');
         this.$backdrop = (this.$el).find('.modal-backdrop');
+        this.$notification = (this.$el).find('.notification-modal');
         
         // append body
         this.$body.html(this.options.body);
@@ -120,6 +161,47 @@
         this.$buttons.find('#' + String(name).toLowerCase()).prop('disabled', true);
     },
     
+    // hide buttons
+    hideButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).hide();
+    },
+    // show buttons
+    showButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).show();
+    },
+
+    // show notification
+    showNotification : function(message, duration, type) {
+        // defaults
+        var duration = typeof duration !== 'undefined' ? duration : 1500;
+        // var bgColor = typeof bgColor !== 'undefined' ? bgColor : "#F4E0E1";
+        // var txtColor = typeof txtColor !== 'undefined' ? txtColor : "#A42732";
+        var bgColor;
+        var txtColor;
+
+        if (type === 'error'){
+            bgColor = '#f4e0e1';
+            txtColor = '#a42732';
+        // } else if (type === 'success'){
+        } else { // success is default
+            bgColor = '#e1f4e0';
+            txtColor = '#32a427'; 
+        }
+
+        var HTMLmessage = "<div class='notification-message' style='text-align:center; line-height:16px; '> " + message + " </div>";
+        this.$notification.html("<div id='notification-bar' style='display:none; float: right; height: 16px; width:100%; background-color: " + bgColor + "; z-index: 100; color: " + txtColor + ";border-bottom: 1px solid " + txtColor + ";'>" + HTMLmessage + "</div>");
+
+        var self = this;
+        
+        /*animate the bar*/
+        $('#notification-bar').slideDown(function() {
+            setTimeout(function() {
+                $('#notification-bar').slideUp(function() {self.$notification.html('');});
+            }, duration);
+        });
+
+    },
+
     // returns scroll top for body element
     scrollTop: function()
     {
@@ -139,7 +221,8 @@
                             '<div class="modal-header">' +
                                 '<button type="button" class="close" style="display: none;">×</button>' +
                                 '<h4 class="title">' + title + '</h4>' +
-                            '</div>' +
+                                '<span class="notification-modal"></span>' + 
+                                '</div>' +
                             '<div class="modal-body"></div>' +
                             '<div class="modal-footer">' +
                                 '<div class="buttons" style="float: right;"></div>' +
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 static/scripts/libs/bootstrap.js
--- a/static/scripts/libs/bootstrap.js
+++ b/static/scripts/libs/bootstrap.js
@@ -575,3 +575,158 @@
   }
 
 }(window.jQuery);
+
+/* ========================================================================
+ * Bootstrap: dropdown.js v3.0.0
+ * http://twbs.github.com/bootstrap/javascript.html#dropdowns
+ * ========================================================================
+ * Copyright 2012 Twitter, Inc.
+ *
+ * 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.
+ * ======================================================================== */
+
+
++function ($) { "use strict";
+
+  // DROPDOWN CLASS DEFINITION
+  // =========================
+
+  var backdrop = '.dropdown-backdrop'
+  var toggle   = '[data-toggle=dropdown]'
+  var Dropdown = function (element) {
+    var $el = $(element).on('click.bs.dropdown', this.toggle)
+  }
+
+  Dropdown.prototype.toggle = function (e) {
+    var $this = $(this)
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    clearMenus()
+
+    if (!isActive) {
+      if ('ontouchstart' in document.documentElement && !$parent.closest('.navbar-nav').length) {
+        // if mobile we we use a backdrop because click events don't delegate
+        $('<div class="dropdown-backdrop"/>').insertAfter($(this)).on('click', clearMenus)
+      }
+
+      $parent.trigger(e = $.Event('show.bs.dropdown'))
+
+      if (e.isDefaultPrevented()) return
+
+      $parent
+        .toggleClass('open')
+        .trigger('shown.bs.dropdown')
+
+      $this.focus()
+    }
+
+    return false
+  }
+
+  Dropdown.prototype.keydown = function (e) {
+    if (!/(38|40|27)/.test(e.keyCode)) return
+
+    var $this = $(this)
+
+    e.preventDefault()
+    e.stopPropagation()
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    if (!isActive || (isActive && e.keyCode == 27)) {
+      if (e.which == 27) $parent.find(toggle).focus()
+      return $this.click()
+    }
+
+    var $items = $('[role=menu] li:not(.divider):visible a', $parent)
+
+    if (!$items.length) return
+
+    var index = $items.index($items.filter(':focus'))
+
+    if (e.keyCode == 38 && index > 0)                 index--                        // up
+    if (e.keyCode == 40 && index < $items.length - 1) index++                        // down
+    if (!~index)                                      index=0
+
+    $items.eq(index).focus()
+  }
+
+  function clearMenus() {
+    $(backdrop).remove()
+    $(toggle).each(function (e) {
+      var $parent = getParent($(this))
+      if (!$parent.hasClass('open')) return
+      $parent.trigger(e = $.Event('hide.bs.dropdown'))
+      if (e.isDefaultPrevented()) return
+      $parent.removeClass('open').trigger('hidden.bs.dropdown')
+    })
+  }
+
+  function getParent($this) {
+    var selector = $this.attr('data-target')
+
+    if (!selector) {
+      selector = $this.attr('href')
+      selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
+    }
+
+    var $parent = selector && $(selector)
+
+    return $parent && $parent.length ? $parent : $this.parent()
+  }
+
+
+  // DROPDOWN PLUGIN DEFINITION
+  // ==========================
+
+  var old = $.fn.dropdown
+
+  $.fn.dropdown = function (option) {
+    return this.each(function () {
+      var $this = $(this)
+      var data  = $this.data('dropdown')
+
+      if (!data) $this.data('dropdown', (data = new Dropdown(this)))
+      if (typeof option == 'string') data[option].call($this)
+    })
+  }
+
+  $.fn.dropdown.Constructor = Dropdown
+
+
+  // DROPDOWN NO CONFLICT
+  // ====================
+
+  $.fn.dropdown.noConflict = function () {
+    $.fn.dropdown = old
+    return this
+  }
+
+
+  // APPLY TO STANDARD DROPDOWN ELEMENTS
+  // ===================================
+
+  $(document)
+    .on('click.bs.dropdown.data-api', clearMenus)
+    .on('click.bs.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
+    .on('click.bs.dropdown.data-api'  , toggle, Dropdown.prototype.toggle)
+    .on('keydown.bs.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown)
+
+}(window.jQuery);
\ No newline at end of file
diff -r 206055969b7cedcbfcac36c19428b9547cd5cb5d -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 static/style/src/less/library.less
--- a/static/style/src/less/library.less
+++ b/static/style/src/less/library.less
@@ -29,6 +29,50 @@
     background-color: @table-bg-accent;
 }
 
+
+tr.light td
+{
+    background-color: white;
+    color: black;
+}
+tr.light:hover td
+{
+    background-color: #f5f5f5;
+    color: #8389a1;
+}
+tr.dark td
+{
+    background-color: #8389a1;
+    color: white;
+}
+tr.dark:hover td
+{
+    background-color: #bbbfd0;
+    color: white;
+}
+a.dark:hover
+{
+    color: yellow;
+    // text-decoration: none;
+}
+a.dark
+{
+    color: white;
+    // text-decoration: none;
+}
+th.button_heading
+{
+    width: 7em;
+}
+div.library_breadcrumb{
+    padding-top: 0.8em;
+    padding-bottom: 0.8em;
+}
+div.library_breadcrumb a:hover{
+    color:green;
+}
+
+
 img.expanderIcon {
     padding-right: 4px;
 }
https://bitbucket.org/galaxy/galaxy-central/commits/caba1617084b/
Changeset:   caba1617084b
User:        martenson
Date:        2013-12-11 20:24:58
Summary:     cleaning zip logic, jmchilton pull rq #208
Affected #:  1 file
diff -r ff102054eec10f7bd1ad9f9663e1884dc7c75223 -r caba1617084b4a0d38f1ebce970dda675755cd1d lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- a/lib/galaxy/webapps/galaxy/api/lda_datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -21,42 +21,6 @@
 import logging
 log = logging.getLogger( __name__ )
 
-# Test for available compression types
-# tmpd = tempfile.mkdtemp()
-# comptypes = []
-# for comptype in ( 'gz', 'bz2' ):
-#     tmpf = os.path.join( tmpd, 'compression_test.tar.' + comptype )
-#     try:
-#         archive = tarfile.open( tmpf, 'w:' + comptype )
-#         archive.close()
-#         comptypes.append( comptype )
-#     except tarfile.CompressionError:
-#         log.exception( "Compression error when testing %s compression.  This option will be disabled for library downloads." % comptype )
-#     try:
-#         os.unlink( tmpf )
-#     except OSError:
-#         pass
-ziptype = '32'
-# tmpf = os.path.join( tmpd, 'compression_test.zip' )
-# try:
-#     archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
-#     archive.close()
-#     comptypes.append( 'zip' )
-#     ziptype = '64'
-# except RuntimeError:
-#     log.exception( "Compression error when testing zip compression. This option will be disabled for library downloads." )
-# except (TypeError, zipfile.LargeZipFile):
-#     # ZIP64 is only in Python2.5+.  Remove TypeError when 2.4 support is dropped
-#     log.warning( 'Max zip file size is 2GB, ZIP64 not supported' )
-#     comptypes.append( 'zip' )
-# try:
-#     os.unlink( tmpf )
-# except OSError:
-#     pass
-# os.rmdir( tmpd )
-
-
-
 class DatasetsController( BaseAPIController, UsesLibraryMixinItems ):
 
     @web.expose_api
@@ -77,13 +41,13 @@
             rval = "Error in dataset API at listing contents: " + str( e )
             log.error( rval + ": %s" % str(e), exc_info=True )
             trans.response.status = 500
-        
+
         rval['id'] = trans.security.encode_id(rval['id']);
         rval['ldda_id'] = trans.security.encode_id(rval['ldda_id']);
         rval['folder_id'] = 'f' + trans.security.encode_id(rval['folder_id'])
-        
+
         return rval
-    
+
     @web.expose
     def download( self, trans, format, **kwd ):
         """
@@ -94,9 +58,9 @@
         lddas = []
 #         is_admin = trans.user_is_admin()
 #         current_user_roles = trans.get_current_user_roles()
-        
+
         datasets_to_download = kwd['ldda_ids%5B%5D']
-        
+
         if ( datasets_to_download != None ):
             datasets_to_download = util.listify( datasets_to_download )
             for dataset_id in datasets_to_download:
@@ -107,7 +71,7 @@
                 except:
                     ldda = None
                     message += "Invalid library dataset id (%s) specified.  " % str( dataset_id )
-                    
+
         if format in [ 'zip','tgz','tbz' ]:
                 error = False
                 killme = string.punctuation + string.whitespace
@@ -119,14 +83,10 @@
                         tmpd = tempfile.mkdtemp()
                         util.umask_fix_perms( tmpd, trans.app.config.umask, 0777, self.app.config.gid )
                         tmpf = os.path.join( tmpd, 'library_download.' + format )
-                        if ziptype == '64' and trans.app.config.upstream_gzip:
+                        if trans.app.config.upstream_gzip:
                             archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED, True )
-                        elif ziptype == '64':
+                        else:
                             archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
-                        elif trans.app.config.upstream_gzip:
-                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED )
-                        else:
-                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED )
                         archive.add = lambda x, y: archive.write( x, y.encode('CP437') )
                     elif format == 'tgz':
                         if trans.app.config.upstream_gzip:
@@ -238,4 +198,4 @@
                 except:
                     return 'This dataset contains no content'
         else:
-            return 'Wrong format';
\ No newline at end of file
+            return 'Wrong format';
https://bitbucket.org/galaxy/galaxy-central/commits/c653272fdbb3/
Changeset:   c653272fdbb3
User:        martenson
Date:        2013-12-11 21:31:20
Summary:     trimming whitespace, import library.less into base.less
Affected #:  2 files
diff -r caba1617084b4a0d38f1ebce970dda675755cd1d -r c653272fdbb3e2fcc8e5dd83b2c71df8bd9b5c2a static/scripts/galaxy.library.js
--- a/static/scripts/galaxy.library.js
+++ b/static/scripts/galaxy.library.js
@@ -7,16 +7,6 @@
 var library_router  = null;
 var responses = [];
 
-
-// load required libraries
-// require([
-//     // load js libraries
-//     'utils/galaxy.css',
-//     ], function(css){
-//         // load css
-//         css.load_file("static/style/library.css");
-// });
-
 // dependencies
 define(["galaxy.modal", "galaxy.masthead", "utils/galaxy.utils"], function(mod_modal, mod_masthead, mod_utils) {
 
@@ -42,7 +32,7 @@
 
     // FOLDER
     var Folder = Backbone.Collection.extend({
-      model: Item      
+      model: Item
   })
 
     // CONTAINER for folder contents (folders, items and metadata).
@@ -78,7 +68,7 @@
     });
 
     //ROUTER
-    var LibraryRouter = Backbone.Router.extend({    
+    var LibraryRouter = Backbone.Router.extend({
         routes: {
             "" :                 "libraries",
             "folders/:id" :      "folder_content",
@@ -98,26 +88,26 @@
     // progress percentage
     progress: 0,
     // progress rate per one item
-    progressStep: 1, 
+    progressStep: 1,
     // last selected history in modal for UX
     lastSelectedHistory: '',
     // self modal
     modal : null,
     // loaded folders
     folders : null,
-    
+
     // initialize
     initialize : function(){
         this.folders = [];
         this.queue = jQuery.Deferred();
         this.queue.resolve();
-    },    
+    },
 
 // MMMMMMMMMMMMMMMMMM
 // === TEMPLATES ====
 // MMMMMMMMMMMMMMMMMM
 
-    // set up 
+    // set up
     templateFolder : function (){
         var tmpl_array = [];
 
@@ -127,12 +117,12 @@
 
         // TOOLBAR
         tmpl_array.push('<div id="library_folder_toolbar" >');
-        tmpl_array.push('   <button title="Create New Folder" id="toolbtn_create_folder" class="btn btn-primary" type="button"><span class="fa fa-icon-plus"></span><span class="fa fa-icon-folder-close"></span> folder</button>');
-        tmpl_array.push('   <button id="toolbtn_bulk_import" class="btn btn-primary" style="display: none; margin-left: 0.5em;" type="button"><span class="fa fa-icon-external-link"></span> to history</button>');
-        
+        tmpl_array.push('   <button title="Create New Folder" id="toolbtn_create_folder" class="btn btn-primary" type="button"><span class="fa fa-plus"></span><span class="fa fa-folder-close"></span> folder</button>');
+        tmpl_array.push('   <button id="toolbtn_bulk_import" class="btn btn-primary" style="display: none; margin-left: 0.5em;" type="button"><span class="fa fa-external-link"></span> to history</button>');
+
         tmpl_array.push('   <div id="toolbtn_dl" class="btn-group" style="margin-left: 0.5em; display: none; ">');
         tmpl_array.push('       <button id="drop_toggle" type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown">');
-        tmpl_array.push('       <span class="fa fa-icon-download"></span> download <span class="caret"></span>');
+        tmpl_array.push('       <span class="fa fa-download"></span> download <span class="caret"></span>');
         tmpl_array.push('       </button>');
         tmpl_array.push('       <ul class="dropdown-menu" role="menu">');
         tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tgz">.tar.gz</a></li>');
@@ -166,20 +156,20 @@
         tmpl_array.push('       <th>date</th>');
         tmpl_array.push('   </thead>');
         tmpl_array.push('   <tbody>');
-        tmpl_array.push('       <td></td>');        
-        tmpl_array.push('       <td><button title="Go to parent folder" type="button" data-id="<%- upper_folder_id %>" class="btn_open_folder btn btn-default btn-xs">');        
-        tmpl_array.push('       <span class="fa fa-icon-arrow-up"></span> .. go up</td>');        
-        tmpl_array.push('       <td></td>');        
-        tmpl_array.push('       <td></td>');        
-        tmpl_array.push('       <td></td>');        
-        tmpl_array.push('       <td></td>');        
-        tmpl_array.push('       </tr>');        
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       <td><button title="Go to parent folder" type="button" data-id="<%- upper_folder_id %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('       <span class="fa fa-arrow-up"></span> .. go up</td>');
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       </tr>');
         tmpl_array.push('       <% _.each(items, function(content_item) { %>');
         tmpl_array.push('       <tr class="folder_row light" id="<%- content_item.id %>">');
         tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>'); // folder
         tmpl_array.push('               <td></td>');
         tmpl_array.push('               <td><button title="Open this folder" type="button" data-id="<%- content_item.id %>" class="btn_open_folder btn btn-default btn-xs">');
-        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <span class="fa fa-folder-open"></span> browse</td>');
         tmpl_array.push('               <td><%- content_item.get("name") %>');
         tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>'); // empty folder
         tmpl_array.push('           <span class="muted">(empty folder)</span>');
@@ -191,7 +181,7 @@
         tmpl_array.push('           <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
         tmpl_array.push('       <td>');
         tmpl_array.push('       <button title="See details of this dataset" type="button" class="library-dataset btn btn-default btn-xs">');
-        tmpl_array.push('       <span class="fa fa-icon-eye-open"></span> details');
+        tmpl_array.push('       <span class="fa fa-eye"></span> details');
         tmpl_array.push('       </button>');
         tmpl_array.push('       </td>');
         tmpl_array.push('           <td><%- content_item.get("name") %></td>'); // dataset
@@ -241,11 +231,11 @@
         tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_data_lines")) %></td>');
         tmpl_array.push('       </tr>');
         tmpl_array.push('       <th scope="row">Comment Lines</th>');
-        tmpl_array.push('           <% if (item.get("metadata_comment_lines") === "") { %>'); //folder        
+        tmpl_array.push('           <% if (item.get("metadata_comment_lines") === "") { %>'); //folder
         tmpl_array.push('               <td scope="row"><%= _.escape(item.get("metadata_comment_lines")) %></td>');
-        tmpl_array.push('           <% } else { %>'); 
-        tmpl_array.push('               <td scope="row">unknown</td>');    
-        tmpl_array.push('           <% } %>'); 
+        tmpl_array.push('           <% } else { %>');
+        tmpl_array.push('               <td scope="row">unknown</td>');
+        tmpl_array.push('           <% } %>');
         tmpl_array.push('       </tr>');
         tmpl_array.push('       <tr>');
         tmpl_array.push('           <th scope="row">Number of Columns</th>');
@@ -280,7 +270,7 @@
         tmpl_array.push('</span>');
 
         return tmpl_array.join('');
-        },   
+        },
 
     templateBulkImportInModal : function(){
         var tmpl_array = [];
@@ -295,7 +285,7 @@
         tmpl_array.push('</span>');
 
         return tmpl_array.join('');
-        },    
+        },
 
       // convert size to nice string
       size_to_string : function (size)
@@ -380,12 +370,12 @@
       showDatasetDetails : function(event){
         // prevent default
         event.preventDefault();
-        
+
 //TODO check whether we already have the data
 
         //load the ID of the row
         var id = $(event.target).parent().parent().attr('id');
-        
+
         //create new item
         var item = new Item();
         var histories = new GalaxyHistories();
@@ -474,12 +464,12 @@
                 self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, 'error');
                 //enable the buttons
                 self.modal.enableButton('Import');
-                self.modal.enableButton('Download');                
+                self.modal.enableButton('Download');
             }
         });
         },
 
-        // select all datasets 
+        // select all datasets
         selectAll : function (event) {
            var selected = event.target.checked;
                  // Iterate each checkbox
@@ -562,7 +552,7 @@
             });
             var progress_bar_tmpl = _.template(this.templateProgressBar(), { history_name : history_name });
             $(this.modal.elMain).find('.modal-body').html(progress_bar_tmpl);
-            
+
             // init the progress bar
             var progressStep = 100 / dataset_ids.length;
             this.initProgress(progressStep);
@@ -610,7 +600,7 @@
             $('.completion_span').text(txt_representation);
         },
 
-      // progress bar 
+      // progress bar
       templateProgressBar : function (){
         var tmpl_array = [];
 
@@ -644,14 +634,14 @@
       // create hidden form and submit through POST to initialize download
       processDownload: function(url, data, method){
         //url and data options required
-        if( url && data ){ 
+        if( url && data ){
                 //data can be string of parameters or array/object
                 data = typeof data == 'string' ? data : $.param(data);
                 //split params into form inputs
                 var inputs = '';
-                $.each(data.split('&'), function(){ 
+                $.each(data.split('&'), function(){
                         var pair = this.split('=');
-                        inputs+='<input type="hidden" name="'+ pair[0] +'" value="'+ pair[1] +'" />'; 
+                        inputs+='<input type="hidden" name="'+ pair[0] +'" value="'+ pair[1] +'" />';
                 });
                 //send request
                 $('<form action="'+ url +'" method="'+ (method||'post') +'">'+inputs+'</form>')
@@ -676,7 +666,7 @@
 
     // initialize
     initialize : function(){
-    },    
+    },
 
     // template
     template_library_list : function (){
@@ -685,7 +675,7 @@
 
         tmpl_array.push('');
         tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
-        tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary icon-file ">New Library</a>');
+        tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary file ">New Library</a>');
         tmpl_array.push('<table class="table table-condensed">');
         tmpl_array.push('   <thead>');
         tmpl_array.push('     <th class="button_heading"></th>');
@@ -698,7 +688,7 @@
         tmpl_array.push('       <% _.each(libraries, function(library) { %>');
         tmpl_array.push('           <tr>');
         tmpl_array.push('               <td><button title="Open this library" type="button" data-id="<%- library.get("root_folder_id") %>" class="btn_open_folder btn btn-default btn-xs">');
-        tmpl_array.push('               <span class="fa fa-icon-folder-open"></span> browse</td>');
+        tmpl_array.push('               <span class="fa fa-folder-open"></span> browse</td>');
         tmpl_array.push('               <td><%- library.get("name") %></td>');
         tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
         tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
@@ -707,7 +697,7 @@
         tmpl_array.push('       <% }); %>');
         tmpl_array.push('   </tbody>');
         tmpl_array.push('</table>');
-        
+
         tmpl_array.push('</div>');
         return tmpl_array.join('');
     },
@@ -735,7 +725,7 @@
     show_library_modal : function (event){
         event.preventDefault();
         event.stopPropagation();
-        
+
         // create modal
             var self = this;
             this.modal = new mod_modal.GalaxyModal(
@@ -747,7 +737,7 @@
                     'Close'  : function() {self.modal.hide()}
                 }
             });
-        
+
         // show prepared modal
         this.modal.show();
     },
@@ -828,7 +818,7 @@
         library_router.on('route:folder_content', function(id) {
           // render folder's contents
           folderContentView.render({id: id});
-      });    
+      });
 
        library_router.on('route:download', function(folder_id, format) {
           // send download stream
@@ -844,9 +834,9 @@
             library_router.navigate('folders/' + folder_id, {trigger: false, replace: true});
           }
 
-      });      
+      });
 
-Backbone.history.start();   
+Backbone.history.start();
 
 return this
 }
diff -r caba1617084b4a0d38f1ebce970dda675755cd1d -r c653272fdbb3e2fcc8e5dd83b2c71df8bd9b5c2a static/style/src/less/base.less
--- a/static/style/src/less/base.less
+++ b/static/style/src/less/base.less
@@ -23,6 +23,7 @@
 // galaxy sub-components
 @import "frame.less";
 @import "upload.less";
+@import "library.less";
 
 // Mixins
 
https://bitbucket.org/galaxy/galaxy-central/commits/267dcfd58a60/
Changeset:   267dcfd58a60
User:        martenson
Date:        2013-12-12 20:21:09
Summary:     added library notifications, toaster, can be used outside of libraries as require.js module
Affected #:  7 files
diff -r c653272fdbb3e2fcc8e5dd83b2c71df8bd9b5c2a -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b static/scripts/galaxy.library.js
--- a/static/scripts/galaxy.library.js
+++ b/static/scripts/galaxy.library.js
@@ -8,7 +8,7 @@
 var responses = [];
 
 // dependencies
-define(["galaxy.modal", "galaxy.masthead", "utils/galaxy.utils"], function(mod_modal, mod_masthead, mod_utils) {
+define(["galaxy.modal", "galaxy.masthead", "utils/galaxy.utils", "libs/toastr"], function(mod_modal, mod_masthead, mod_utils, mod_toastr) {
 
 // MMMMMMMMMMMMMMM
 // === Models ====
@@ -706,18 +706,35 @@
     render: function () {
         //hack to show scrollbars
         $("#center").css('overflow','auto');
+        
         var that = this;
-        // if (typeof libraries === "undefined") {
-          libraries = new Libraries();
-        // }
+        libraries = new Libraries();
+
         libraries.fetch({
           success: function (libraries) {
             var template = _.template(that.template_library_list(), { libraries : libraries.models });
             that.$el.html(template);
-        }
+          },
+          error: function(model, response){
+            if (response.statusCode().status === 403){
+                //show notification
+                // var toast = mod_toastr;
+                mod_toastr.error('Please log in first. Redirecting to login page in 3s.');   
+                setTimeout(that.redirectToLogin, 3000);
+            } else {
+                library_router.navigate('#', {trigger: true, replace: true});
+            }
+          }
     })
     },
 
+    redirectToSplash: function(){
+        window.location = '../';
+    },    
+    redirectToLogin: function(){
+        window.location = '/user/login';
+    },
+
     // own modal
     modal : null,
 
diff -r c653272fdbb3e2fcc8e5dd83b2c71df8bd9b5c2a -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -172,10 +172,7 @@
 
     // show notification
     showNotification : function(message, duration, type) {
-        // defaults
-        var duration = typeof duration !== 'undefined' ? duration : 1500;
-        // var bgColor = typeof bgColor !== 'undefined' ? bgColor : "#F4E0E1";
-        // var txtColor = typeof txtColor !== 'undefined' ? txtColor : "#A42732";
+        var duration = typeof duration !== 'undefined' ? duration : 3000;
         var bgColor;
         var txtColor;
 
diff -r c653272fdbb3e2fcc8e5dd83b2c71df8bd9b5c2a -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b static/scripts/libs/toastr.js
--- /dev/null
+++ b/static/scripts/libs/toastr.js
@@ -0,0 +1,307 @@
+/*
+ * Toastr
+ * Version 2.0.1
+ * Copyright 2012 John Papa and Hans Fjällemark.  
+ * All Rights Reserved.  
+ * Use, reproduction, distribution, and modification of this code is subject to the terms and 
+ * conditions of the MIT license, available at http://www.opensource.org/licenses/mit-license.php
+ *
+ * Author: John Papa and Hans Fjällemark
+ * Project: https://github.com/CodeSeven/toastr
+ */
+; (function (define) {
+	define([], function () {
+		var $ = jQuery;
+		return (function () {
+			var version = '2.0.1';
+			var $container;
+			var listener;
+			var toastId = 0;
+			var toastType = {
+				error: 'error',
+				info: 'info',
+				success: 'success',
+				warning: 'warning'
+			};
+
+			var toastr = {
+				clear: clear,
+				error: error,
+				getContainer: getContainer,
+				info: info,
+				options: {},
+				subscribe: subscribe,
+				success: success,
+				version: version,
+				warning: warning
+			};
+
+			return toastr;
+
+			//#region Accessible Methods
+			function error(message, title, optionsOverride) {
+				return notify({
+					type: toastType.error,
+					iconClass: getOptions().iconClasses.error,
+					message: message,
+					optionsOverride: optionsOverride,
+					title: title
+				});
+			}
+
+			function info(message, title, optionsOverride) {
+				return notify({
+					type: toastType.info,
+					iconClass: getOptions().iconClasses.info,
+					message: message,
+					optionsOverride: optionsOverride,
+					title: title
+				});
+			}
+
+			function subscribe(callback) {
+				listener = callback;
+			}
+
+			function success(message, title, optionsOverride) {
+				return notify({
+					type: toastType.success,
+					iconClass: getOptions().iconClasses.success,
+					message: message,
+					optionsOverride: optionsOverride,
+					title: title
+				});
+			}
+
+			function warning(message, title, optionsOverride) {
+				return notify({
+					type: toastType.warning,
+					iconClass: getOptions().iconClasses.warning,
+					message: message,
+					optionsOverride: optionsOverride,
+					title: title
+				});
+			}
+
+			function clear($toastElement) {
+				var options = getOptions();
+				if (!$container) { getContainer(options); }
+				if ($toastElement && $(':focus', $toastElement).length === 0) {
+					$toastElement[options.hideMethod]({
+						duration: options.hideDuration,
+						easing: options.hideEasing,
+						complete: function () { removeToast($toastElement); }
+					});
+					return;
+				}
+				if ($container.children().length) {
+					$container[options.hideMethod]({
+						duration: options.hideDuration,
+						easing: options.hideEasing,
+						complete: function () { $container.remove(); }
+					});
+				}
+			}
+			//#endregion
+
+			//#region Internal Methods
+
+			function getDefaults() {
+				return {
+					tapToDismiss: true,
+					toastClass: 'toast',
+					containerId: 'toast-container',
+					debug: false,
+
+					showMethod: 'fadeIn', //fadeIn, slideDown, and show are built into jQuery
+					showDuration: 300,
+					showEasing: 'swing', //swing and linear are built into jQuery
+					onShown: undefined,
+					hideMethod: 'fadeOut',
+					hideDuration: 1000,
+					hideEasing: 'swing',
+					onHidden: undefined,
+
+					extendedTimeOut: 1000,
+					iconClasses: {
+						error: 'toast-error',
+						info: 'toast-info',
+						success: 'toast-success',
+						warning: 'toast-warning'
+					},
+					iconClass: 'toast-info',
+					positionClass: 'toast-top-right',
+					timeOut: 5000, // Set timeOut and extendedTimeout to 0 to make it sticky
+					titleClass: 'toast-title',
+					messageClass: 'toast-message',
+					target: 'body',
+					closeHtml: '<button>×</button>',
+					newestOnTop: true
+				};
+			}
+
+			function publish(args) {
+				if (!listener) {
+					return;
+				}
+				listener(args);
+			}
+
+			function notify(map) {
+				var
+					options = getOptions(),
+					iconClass = map.iconClass || options.iconClass;
+
+				if (typeof (map.optionsOverride) !== 'undefined') {
+					options = $.extend(options, map.optionsOverride);
+					iconClass = map.optionsOverride.iconClass || iconClass;
+				}
+
+				toastId++;
+
+				$container = getContainer(options);
+				var
+					intervalId = null,
+					$toastElement = $('<div/>'),
+					$titleElement = $('<div/>'),
+					$messageElement = $('<div/>'),
+					$closeElement = $(options.closeHtml),
+					response = {
+						toastId: toastId,
+						state: 'visible',
+						startTime: new Date(),
+						options: options,
+						map: map
+					};
+
+				if (map.iconClass) {
+					$toastElement.addClass(options.toastClass).addClass(iconClass);
+				}
+
+				if (map.title) {
+					$titleElement.append(map.title).addClass(options.titleClass);
+					$toastElement.append($titleElement);
+				}
+
+				if (map.message) {
+					$messageElement.append(map.message).addClass(options.messageClass);
+					$toastElement.append($messageElement);
+				}
+
+				if (options.closeButton) {
+					$closeElement.addClass('toast-close-button');
+					$toastElement.prepend($closeElement);
+				}
+
+				$toastElement.hide();
+				if (options.newestOnTop) {
+					$container.prepend($toastElement);
+				} else {
+					$container.append($toastElement);
+				}
+
+
+				$toastElement[options.showMethod](
+					{ duration: options.showDuration, easing: options.showEasing, complete: options.onShown }
+				);
+				if (options.timeOut > 0) {
+					intervalId = setTimeout(hideToast, options.timeOut);
+				}
+
+				$toastElement.hover(stickAround, delayedhideToast);
+				if (!options.onclick && options.tapToDismiss) {
+					$toastElement.click(hideToast);
+				}
+				if (options.closeButton && $closeElement) {
+					$closeElement.click(function (event) {
+						event.stopPropagation();
+						hideToast(true);
+					});
+				}
+
+				if (options.onclick) {
+					$toastElement.click(function () {
+						options.onclick();
+						hideToast();
+					});
+				}
+
+				publish(response);
+
+				if (options.debug && console) {
+					console.log(response);
+				}
+
+				return $toastElement;
+
+				function hideToast(override) {
+					if ($(':focus', $toastElement).length && !override) {
+						return;
+					}
+					return $toastElement[options.hideMethod]({
+						duration: options.hideDuration,
+						easing: options.hideEasing,
+						complete: function () {
+							removeToast($toastElement);
+							if (options.onHidden) {
+								options.onHidden();
+							}
+							response.state = 'hidden';
+							response.endTime = new Date(),
+							publish(response);
+						}
+					});
+				}
+
+				function delayedhideToast() {
+					if (options.timeOut > 0 || options.extendedTimeOut > 0) {
+						intervalId = setTimeout(hideToast, options.extendedTimeOut);
+					}
+				}
+
+				function stickAround() {
+					clearTimeout(intervalId);
+					$toastElement.stop(true, true)[options.showMethod](
+						{ duration: options.showDuration, easing: options.showEasing }
+					);
+				}
+			}
+			function getContainer(options) {
+				if (!options) { options = getOptions(); }
+				$container = $('#' + options.containerId);
+				if ($container.length) {
+					return $container;
+				}
+				$container = $('<div/>')
+					.attr('id', options.containerId)
+					.addClass(options.positionClass);
+				$container.appendTo($(options.target));
+				return $container;
+			}
+
+			function getOptions() {
+				return $.extend({}, getDefaults(), toastr.options);
+			}
+
+			function removeToast($toastElement) {
+				if (!$container) { $container = getContainer(); }
+				if ($toastElement.is(':visible')) {
+					return;
+				}
+				$toastElement.remove();
+				$toastElement = null;
+				if ($container.children().length === 0) {
+					$container.remove();
+				}
+			}
+			//#endregion
+
+		})();
+	});
+}(typeof define === 'function' && define.amd ? define : function (deps, factory) {
+	if (typeof module !== 'undefined' && module.exports) { //Node
+		module.exports = factory(require(deps[0]));
+	} else {
+		window['toastr'] = factory(window['jQuery']);
+	}
+}));
diff -r c653272fdbb3e2fcc8e5dd83b2c71df8bd9b5c2a -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b static/scripts/utils/galaxy.uploadbox.js
--- a/static/scripts/utils/galaxy.uploadbox.js
+++ b/static/scripts/utils/galaxy.uploadbox.js
@@ -352,4 +352,4 @@
             'compatible'    : compatible
         };
     }
-})(jQuery);
\ No newline at end of file
+})(jQuery);
diff -r c653272fdbb3e2fcc8e5dd83b2c71df8bd9b5c2a -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b static/style/Gruntfile.js
--- a/static/style/Gruntfile.js
+++ b/static/style/Gruntfile.js
@@ -6,7 +6,7 @@
   var theme = grunt.option( 'theme', 'blue' );
 
   var out = 'blue'
-  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'trackster' ];
+  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'trackster', 'toastr' ];
 
   var _ = grunt.util._; 
   var fmt = _.sprintf; 
diff -r c653272fdbb3e2fcc8e5dd83b2c71df8bd9b5c2a -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b static/style/src/less/base.less
--- a/static/style/src/less/base.less
+++ b/static/style/src/less/base.less
@@ -24,6 +24,7 @@
 @import "frame.less";
 @import "upload.less";
 @import "library.less";
+@import "toastr.less";
 
 // Mixins
 
diff -r c653272fdbb3e2fcc8e5dd83b2c71df8bd9b5c2a -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b static/style/src/less/toastr.less
--- /dev/null
+++ b/static/style/src/less/toastr.less
@@ -0,0 +1,232 @@
+// Mix-ins
+.borderRadius(@radius) {
+	-moz-border-radius: @radius;
+	-webkit-border-radius: @radius;
+	border-radius: @radius;
+}
+
+.boxShadow(@boxShadow) {
+	-moz-box-shadow: @boxShadow;
+	-webkit-box-shadow: @boxShadow;
+	box-shadow: @boxShadow;
+}
+
+.opacity(@opacity) {
+	@opacityPercent: @opacity * 100;
+	opacity: @opacity;
+	-ms-filter: ~"progid:DXImageTransform.Microsoft.Alpha(Opacity=@{opacityPercent})";
+	filter: ~"alpha(opacity=@{opacityPercent})";
+}
+
+.wordWrap(@wordWrap: break-word) {
+	-ms-word-wrap: @wordWrap;
+	word-wrap: @wordWrap;
+}
+
+// Variables
+@black: #000000;
+@grey: #999999;
+@light-grey: #CCCCCC;
+@white: #FFFFFF;
+@near-black: #030303;
+@green: #51A351;
+@red: #BD362F;
+@blue: #2F96B4;
+@orange: #F89406;
+
+// Styles
+.toast-title {
+	font-weight: bold;
+}
+
+.toast-message {
+	.wordWrap();
+
+	a,
+	label {
+		color: @white;
+	}
+
+		a:hover {
+			color: @light-grey;
+			text-decoration: none;
+		}
+}
+
+.toast-close-button {
+	position: relative;
+	right: -0.3em;
+	top: -0.3em;
+	float: right;
+	font-size: 20px;
+	font-weight: bold;
+	color: @white;
+	-webkit-text-shadow: 0 1px 0 rgba(255,255,255,1);
+	text-shadow: 0 1px 0 rgba(255,255,255,1);
+	.opacity(0.8);
+
+	&:hover,
+	&:focus {
+		color: @black;
+		text-decoration: none;
+		cursor: pointer;
+		.opacity(0.4);
+	}
+}
+
+/*Additional properties for button version
+ iOS requires the button element instead of an anchor tag.
+ If you want the anchor version, it requires `href="#"`.*/
+button.toast-close-button {
+	padding: 0;
+	cursor: pointer;
+	background: transparent;
+	border: 0;
+	-webkit-appearance: none;
+}
+
+//#endregion
+
+.toast-top-full-width {
+	top: 0;
+	right: 0;
+	width: 100%;
+}
+
+.toast-bottom-full-width {
+	bottom: 0;
+	right: 0;
+	width: 100%;
+}
+
+.toast-top-left {
+	top: 12px;
+	left: 12px;
+}
+
+.toast-top-right {
+	top: 12px;
+	right: 12px;
+}
+
+.toast-bottom-right {
+	right: 12px;
+	bottom: 12px;
+}
+
+.toast-bottom-left {
+	bottom: 12px;
+	left: 12px;
+}
+
+#toast-container {
+	position: fixed;
+	z-index: 999999;
+
+	* {
+		-moz-box-sizing: border-box;
+		-webkit-box-sizing: border-box;
+		box-sizing: border-box;
+	}
+
+	> div {
+		margin: 4em 0 6px;
+		padding: 15px 15px 15px 50px;
+		width: 300px;
+		.borderRadius(3px 3px 3px 3px);
+		background-position: 15px center;
+		background-repeat: no-repeat;
+		.boxShadow(0 0 12px @grey);
+		color: @white;
+		.opacity(0.8);
+	}
+
+	> :hover {
+		.boxShadow(0 0 12px @black);
+		.opacity(1);
+		cursor: pointer;
+	}
+
+	> .toast-info {
+		background-image: url("") !important;
+	}
+
+	> .toast-error {
+		background-image: url("") !important;
+	}
+
+	> .toast-success {
+		background-image: url("") !important;
+	}
+
+	> .toast-warning {
+		background-image: url("") !important;
+	}
+
+	/*overrides*/
+	&.toast-top-full-width > div,
+	&.toast-bottom-full-width > div {
+		width: 96%;
+		margin: auto;
+	}
+}
+
+.toast {
+	background-color: @near-black;
+}
+
+.toast-success {
+	background-color: @green;
+}
+
+.toast-error {
+	background-color: @red;
+}
+
+.toast-info {
+	background-color: @blue;
+}
+
+.toast-warning {
+	background-color: @orange;
+}
+
+/*Responsive Design*/
+
+@media all and (max-width: 240px) {
+	#toast-container {
+
+		> div {
+			padding: 8px 8px 8px 50px;
+			width: 11em;
+		}
+
+		& .toast-close-button {
+			right: -0.2em;
+			top: -0.2em;
+		}
+	}
+}
+
+@media all and (min-width: 241px) and (max-width: 480px) {
+	#toast-container {
+		> div {
+			padding: 8px 8px 8px 50px;
+			width: 18em;
+		}
+
+		& .toast-close-button {
+			right: -0.2em;
+			top: -0.2em;
+		}
+	}
+}
+
+@media all and (min-width: 481px) and (max-width: 768px) {
+	#toast-container {
+		> div {
+			padding: 15px 15px 15px 50px;
+			width: 25em;
+		}
+	}
+}
https://bitbucket.org/galaxy/galaxy-central/commits/9e7cd2749bde/
Changeset:   9e7cd2749bde
User:        martenson
Date:        2013-12-13 20:33:06
Summary:     rewrite of galaxy.library to use global notifications; removed previous notification engine from galaxy.modal.js;
Affected #:  3 files
diff -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b -r 9e7cd2749bdeb36bab9a7c883d95a877f9eaa2ca lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -144,38 +144,6 @@
                             parent_resources=dict( member_name='datatype', collection_name='datatypes' ) )
     #webapp.mapper.connect( 'run_workflow', '/api/workflow/{workflow_id}/library/{library_id}', controller='workflows', action='run', workflow_id=None, library_id=None, conditions=dict(method=["GET"]) )
     webapp.mapper.resource( 'search', 'search', path_prefix='/api' )
-
-    # =======================
-    # ===== LIBRARY API =====
-    # =======================
-    
-    # The /folders section is experimental at this point:
-    log.debug( "app.config.api_folders: %s" % app.config.api_folders )
-
-    webapp.mapper.connect( 'show_lda_item', '/api/libraries/datasets/:id', controller='lda_datasets', action='show', conditions=dict(method=["GET"]) )
-    webapp.mapper.connect( 'download_lda_items', '/api/libraries/datasets/download/:format', controller='lda_datasets', action='download', conditions=dict(method=["POST"]) )
-
-    webapp.mapper.resource_with_deleted( 'library', 'libraries', path_prefix='/api' )
-    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
-
-    webapp.mapper.resource( 'content', 'contents',
-                                controller='folder_contents',
-                                name_prefix='folder_',
-                                path_prefix='/api/folders/:folder_id',
-                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
-    webapp.mapper.resource( 'content',
-                                'contents',
-                                controller='library_contents',
-                                name_prefix='library_',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
-    webapp.mapper.resource( 'permission',
-                                'permissions',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
-    _add_item_extended_metadata_controller( webapp,
-                               name_prefix="library_dataset_",
-                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
     
 
     # add as a non-ATOM API call to support the notion of a 'current/working' history unique to the history resource
@@ -195,6 +163,52 @@
     webapp.mapper.connect("workflow_dict", '/api/workflows/{workflow_id}/download', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
     # Preserve the following download route for now for dependent applications  -- deprecate at some point
     webapp.mapper.connect("workflow_dict", '/api/workflows/download/{workflow_id}', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
+
+    # =======================
+    # ===== LIBRARY API =====
+    # =======================
+    
+    webapp.mapper.connect( 'show_lda_item',
+                           '/api/libraries/datasets/:id', 
+                           controller='lda_datasets', 
+                           action='show', 
+                           conditions=dict( method=[ "GET" ] ) )
+
+    webapp.mapper.connect( 'download_lda_items', 
+                           '/api/libraries/datasets/download/:format', 
+                           controller='lda_datasets', 
+                           action='download', 
+                           conditions=dict( method=[ "POST", "GET" ] ) )
+
+    webapp.mapper.resource_with_deleted( 'library',
+                                         'libraries', 
+                                         path_prefix='/api' )
+    webapp.mapper.resource( 'folder', 
+                            'folders', 
+                            path_prefix='/api' )
+
+    webapp.mapper.resource( 'content', 
+                            'contents',
+                            controller='folder_contents',
+                            name_prefix='folder_',
+                            path_prefix='/api/folders/:folder_id',
+                            parent_resources=dict( member_name='folder', collection_name='folders' ) )
+
+    webapp.mapper.resource( 'content',
+                            'contents',
+                            controller='library_contents',
+                            name_prefix='library_',
+                            path_prefix='/api/libraries/:library_id',
+                            parent_resources=dict( member_name='library', collection_name='libraries' ) )
+
+    webapp.mapper.resource( 'permission',
+                            'permissions',
+                            path_prefix='/api/libraries/:library_id',
+                            parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    
+    _add_item_extended_metadata_controller( webapp,
+                               name_prefix="library_dataset_",
+                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
     
     # ====================
     # ===== TOOLSHED =====
diff -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b -r 9e7cd2749bdeb36bab9a7c883d95a877f9eaa2ca static/scripts/galaxy.library.js
--- a/static/scripts/galaxy.library.js
+++ b/static/scripts/galaxy.library.js
@@ -8,7 +8,11 @@
 var responses = [];
 
 // dependencies
-define(["galaxy.modal", "galaxy.masthead", "utils/galaxy.utils", "libs/toastr"], function(mod_modal, mod_masthead, mod_utils, mod_toastr) {
+define([
+    "galaxy.modal", 
+    "galaxy.masthead", 
+    "utils/galaxy.utils", 
+    "libs/toastr"], function(mod_modal, mod_masthead, mod_utils, mod_toastr) {
 
 // MMMMMMMMMMMMMMM
 // === Models ====
@@ -308,7 +312,7 @@
       // event binding
       events: {
             'click #select-all-checkboxes' : 'selectAll',
-            'click .folder_row' : 'selectClicked',
+            'click .folder_row' : 'selectClickedRow',
             'click #toolbtn_bulk_import' : 'modalBulkImport',
             'click #toolbtn_dl' : 'bulkDownload',
             'click .library-dataset' : 'showDatasetDetails',
@@ -350,6 +354,7 @@
               var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id, upper_folder_id: upper_folder_id });
               // var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id });
               that.$el.html(template);
+
             }
           })
         },
@@ -456,12 +461,14 @@
 
             // save the dataset into selected history
             historyItem.save({ content : library_dataset_id, source : 'library' }, { success : function(){
-                self.modal.showNotification('Dataset imported', 3000, 'success');
+                mod_toastr.success('Dataset imported');
+                // self.modal.showNotification('Dataset imported', 3000, 'success');
                 //enable the buttons
                 self.modal.enableButton('Import');
                 self.modal.enableButton('Download');
             }, error : function(){
-                self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, 'error');
+                mod_toastr.error('An error occured! Dataset not imported. Please try again.')
+                // self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, 'error');
                 //enable the buttons
                 self.modal.enableButton('Import');
                 self.modal.enableButton('Download');
@@ -471,37 +478,69 @@
 
         // select all datasets
         selectAll : function (event) {
-           var selected = event.target.checked;
-                 // Iterate each checkbox
-                 $(':checkbox').each(function () { this.checked = selected; });
-                 this.showTools();
-             },
+             var selected = event.target.checked;
+             that = this;
+             // Iterate each checkbox
+             $(':checkbox').each(function () { 
+                this.checked = selected; 
+                $row = $(this.parentElement.parentElement);
+                // Change color of selected/unselected
+                (selected) ? that.makeDarkRow($row) : that.makeWhiteRow($row);
+            });
+             // Show the tools in menu
+             this.checkTools();
+         },
 
-         // click checkbox on folder click
-         selectClicked : function (event) {
-            var checkbox = $("#" + event.target.parentElement.id).find(':checkbox')
-                if (checkbox[0] != undefined) {
-                  if (checkbox[0].checked){
-                    checkbox[0].checked = '';
-                    // $(event.target.parentElement).css('background-color', '').css('color', '');
-                    $(event.target.parentElement).removeClass('dark');
-                    $(event.target.parentElement).find('a').removeClass('dark');
-                    $(event.target.parentElement).addClass('light');
-                    $(event.target.parentElement).find('a').addClass('light');
+         // Check checkbox on row itself or row checkbox click
+         selectClickedRow : function (event) {
+            var checkbox = '';
+            var $row;
+            var source;
+            if (event.target.localName === 'input'){
+                checkbox = event.target;
+                $row = $(event.target.parentElement.parentElement);
+                source = 'input';
+            } else if (event.target.localName === 'td') {
+                checkbox = $("#" + event.target.parentElement.id).find(':checkbox')[0];
+                $row = $(event.target.parentElement);
+                source = 'td';
+            }
+            if (checkbox === '') {event.stopPropagation(); return;} // button in row was clicked
+
+                if (checkbox.checked){
+                    if (source==='td'){
+                        checkbox.checked = '';
+                        this.makeWhiteRow($row);
+                    } else if (source==='input') {
+                        this.makeDarkRow($row);
+                    }
                 } else {
-                    checkbox[0].checked = 'selected';
-                    $(event.target.parentElement).removeClass('light');
-                    $(event.target.parentElement).find('a').removeClass('light');
-                    $(event.target.parentElement).addClass('dark');
-                    $(event.target.parentElement).find('a').addClass('dark');
-                    // $(event.target.parentElement).css('background-color', '#8389a1').css('color', 'white');
+                    if (source==='td'){
+                        checkbox.checked = 'selected';
+                        this.makeDarkRow($row);
+                    } else if (source==='input') {
+                        this.makeWhiteRow($row);
+                    }
                 }
-            }
-            this.showTools();
+                this.checkTools();
+        },
+
+        makeDarkRow: function($row){
+            $row.removeClass('light');
+            $row.find('a').removeClass('light');
+            $row.addClass('dark');
+            $row.find('a').addClass('dark');
+        },
+
+        makeWhiteRow: function($row){
+            $row.removeClass('dark');
+            $row.find('a').removeClass('dark');
+            $row.addClass('light');
+            $row.find('a').addClass('light');
         },
 
         // show toolbar in case something is selected
-        showTools : function(){
+        checkTools : function(){
             var checkedValues = $('#folder_table').find(':checked');
             if(checkedValues.length > 0){
                 $('#toolbtn_bulk_import').show();
@@ -577,7 +616,9 @@
             var self = this;
             var popped_item = history_item_set.pop();
             if (typeof popped_item === "undefined") {
-                self.modal.showNotification('All datasets imported', 3000, 'success');
+                mod_toastr.success('All datasets imported');
+                this.modal.hide();
+                // self.modal.showNotification('All datasets imported', 3000, 'success');
                 // enable button again
                 self.modal.enableButton('Import');
                 return
@@ -646,12 +687,13 @@
                 //send request
                 $('<form action="'+ url +'" method="'+ (method||'post') +'">'+inputs+'</form>')
                 .appendTo('body').submit().remove();
+                mod_toastr.info('Your download will begin soon');
         };
       },
 
       // shows modal for creating folder
       createFolderModal: function(){
-        alert('creating folder');
+        mod_toastr.info('This will create folder...in the future');
       }
 
     });
@@ -704,7 +746,7 @@
 
     // render
     render: function () {
-        //hack to show scrollbars
+        //hack to show scrollbars due to #center element inheritance
         $("#center").css('overflow','auto');
         
         var that = this;
@@ -717,18 +759,16 @@
           },
           error: function(model, response){
             if (response.statusCode().status === 403){
-                //show notification
-                // var toast = mod_toastr;
                 mod_toastr.error('Please log in first. Redirecting to login page in 3s.');   
                 setTimeout(that.redirectToLogin, 3000);
             } else {
-                library_router.navigate('#', {trigger: true, replace: true});
+                mod_toastr.error('An error occured. Please try again.');
             }
           }
     })
     },
 
-    redirectToSplash: function(){
+    redirectToHome: function(){
         window.location = '../';
     },    
     redirectToLogin: function(){
@@ -762,19 +802,23 @@
     // create the new library from modal
     create_new_library_event: function(){
         var libraryDetails = this.serialize_new_library();
-        var valid = this.validate_new_library(libraryDetails);
-        var library = new Library();
-        var self = this;
-        library.save(libraryDetails, {
-          success: function (library) {
-            self.modal.hide();
-            self.clear_library_modal();
-            self.render();
-          },
-          error: function(){
-            self.modal.showNotification('An error occured', 5000, 'error');
-            }
-    });
+        if (this.validate_new_library(libraryDetails)){
+            var library = new Library();
+            var self = this;
+            library.save(libraryDetails, {
+              success: function (library) {
+                self.modal.hide();
+                self.clear_library_modal();
+                self.render();
+                mod_toastr.success('Library created');
+              },
+              error: function(){
+                mod_toastr.error('An error occured :(');
+              }
+            });
+        } else {
+            mod_toastr.error('Library\'s name is missing');
+        }
         return false;
     },
 
@@ -794,14 +838,13 @@
         };
     },
 
-    validate_new_library: function(library){
-
+    // validate new library info
+    validate_new_library: function(libraryDetails){
+        return libraryDetails.name !== '';
     },
 
-
     // template for new library modal
-    template_new_library: function()
-    {
+    template_new_library: function(){
         tmpl_array = [];
 
         tmpl_array.push('<div id="new_library_modal">');
@@ -824,34 +867,29 @@
 
         folderContentView = new FolderContentView();
         galaxyLibraryview = new GalaxyLibraryview();
-
         library_router = new LibraryRouter();
 
         library_router.on('route:libraries', function() {
           // render libraries list
           galaxyLibraryview.render();
-      });
+        });
 
         library_router.on('route:folder_content', function(id) {
           // render folder's contents
           folderContentView.render({id: id});
-      });
+        });
 
        library_router.on('route:download', function(folder_id, format) {
-          // send download stream
-          if (typeof folderContentView === 'undefined'){
-            alert('you cant touch this!');
-          // } else if (folderContentView.modal !== null){
-            // folderContentView.download(folder_id, format);
-          } else if ($('#center').find(':checked').length === 0) { // coming from outside of the library app
+          if ($('#center').find(':checked').length === 0) { 
+            // this happens rarely when there is a server/data error and client gets an actual response instead of an attachment
+            // we don't know what was selected so we can't download again, we redirect to the folder provided
             library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
-            // TODO show message of redirection
           } else {
+            // send download stream
             folderContentView.download(folder_id, format);
             library_router.navigate('folders/' + folder_id, {trigger: false, replace: true});
           }
-
-      });
+        });
 
 Backbone.history.start();
 
diff -r 267dcfd58a60d2de706d7c00c6fb6ec426eaaf5b -r 9e7cd2749bdeb36bab9a7c883d95a877f9eaa2ca static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -26,16 +26,17 @@
         if (options)
             this.create(options);
 
-        this.bindClick(event, self);
+        // Bind the hiding events
+        this.bindEvents(event, self);
     },
 
     // bind the click-to-hide function
-    bindClick: function(event, that) {
+    bindEvents: function(event, that) {
         // bind the ESC key to hide() function
         $(document).on('keyup', function(event){
             if (event.keyCode == 27) { self.hide(); }
         })
-        // bind the click anywhere to hide() function...
+        // bind the 'click anywhere' to hide() function...
         $('html').on('click', function(event){
             that.hide();
         })
@@ -46,26 +47,24 @@
     },
 
     // unbind the click-to-hide function
-    unbindClick: function(event, that){
+    unbindEvents: function(event, that){
         // bind the ESC key to hide() function
         $(document).off('keyup', function(event){
             if (event.keyCode == 27) { that.hide(); }
         })
-        // unbind the click anywhere to hide() function...
+        // unbind the 'click anywhere' to hide() function...
         $('html').off('click', function(event){
             that.hide();
         })
-        // ...but don't hide if the click is on modal content
         $('.modal-content').off('click', function(event){
             event.stopPropagation();
         })
     },
 
-
     // destroy
     destroy : function(){
         this.hide();
-        this.unbindClick();
+        this.unbindEvents();
         $('.modal').remove();
     },
 
@@ -103,6 +102,7 @@
         
         // set flag
         this.visible = false;
+        this.unbindEvents();
     },
     
     // create
@@ -127,7 +127,6 @@
         this.$footer  = (this.$el).find('.modal-footer');
         this.$buttons = (this.$el).find('.buttons');
         this.$backdrop = (this.$el).find('.modal-backdrop');
-        this.$notification = (this.$el).find('.notification-modal');
         
         // append body
         this.$body.html(this.options.body);
@@ -170,35 +169,6 @@
         this.$buttons.find('#' + String(name).toLowerCase()).show();
     },
 
-    // show notification
-    showNotification : function(message, duration, type) {
-        var duration = typeof duration !== 'undefined' ? duration : 3000;
-        var bgColor;
-        var txtColor;
-
-        if (type === 'error'){
-            bgColor = '#f4e0e1';
-            txtColor = '#a42732';
-        // } else if (type === 'success'){
-        } else { // success is default
-            bgColor = '#e1f4e0';
-            txtColor = '#32a427'; 
-        }
-
-        var HTMLmessage = "<div class='notification-message' style='text-align:center; line-height:16px; '> " + message + " </div>";
-        this.$notification.html("<div id='notification-bar' style='display:none; float: right; height: 16px; width:100%; background-color: " + bgColor + "; z-index: 100; color: " + txtColor + ";border-bottom: 1px solid " + txtColor + ";'>" + HTMLmessage + "</div>");
-
-        var self = this;
-        
-        /*animate the bar*/
-        $('#notification-bar').slideDown(function() {
-            setTimeout(function() {
-                $('#notification-bar').slideUp(function() {self.$notification.html('');});
-            }, duration);
-        });
-
-    },
-
     // returns scroll top for body element
     scrollTop: function()
     {
@@ -218,7 +188,6 @@
                             '<div class="modal-header">' +
                                 '<button type="button" class="close" style="display: none;">×</button>' +
                                 '<h4 class="title">' + title + '</h4>' +
-                                '<span class="notification-modal"></span>' + 
                                 '</div>' +
                             '<div class="modal-body"></div>' +
                             '<div class="modal-footer">' +
https://bitbucket.org/galaxy/galaxy-central/commits/0522f7504094/
Changeset:   0522f7504094
User:        martenson
Date:        2013-12-13 21:05:26
Summary:     switch download to GET, removed toaster from grunt file
Affected #:  2 files
diff -r 9e7cd2749bdeb36bab9a7c883d95a877f9eaa2ca -r 0522f7504094bd5be2be97846e695a0c790bf929 static/scripts/galaxy.library.js
--- a/static/scripts/galaxy.library.js
+++ b/static/scripts/galaxy.library.js
@@ -462,13 +462,11 @@
             // save the dataset into selected history
             historyItem.save({ content : library_dataset_id, source : 'library' }, { success : function(){
                 mod_toastr.success('Dataset imported');
-                // self.modal.showNotification('Dataset imported', 3000, 'success');
                 //enable the buttons
                 self.modal.enableButton('Import');
                 self.modal.enableButton('Download');
             }, error : function(){
                 mod_toastr.error('An error occured! Dataset not imported. Please try again.')
-                // self.modal.showNotification('An error occured! Dataset not imported. Please try again later.', 5000, 'error');
                 //enable the buttons
                 self.modal.enableButton('Import');
                 self.modal.enableButton('Download');
@@ -618,7 +616,6 @@
             if (typeof popped_item === "undefined") {
                 mod_toastr.success('All datasets imported');
                 this.modal.hide();
-                // self.modal.showNotification('All datasets imported', 3000, 'success');
                 // enable button again
                 self.modal.enableButton('Import');
                 return
@@ -669,7 +666,7 @@
 
         var url = '/api/libraries/datasets/download/' + format;
         var data = {'ldda_ids' : dataset_ids};
-        this.processDownload(url, data);
+        this.processDownload(url, data, 'get');
       },
 
       // create hidden form and submit through POST to initialize download
diff -r 9e7cd2749bdeb36bab9a7c883d95a877f9eaa2ca -r 0522f7504094bd5be2be97846e695a0c790bf929 static/style/Gruntfile.js
--- a/static/style/Gruntfile.js
+++ b/static/style/Gruntfile.js
@@ -6,7 +6,7 @@
   var theme = grunt.option( 'theme', 'blue' );
 
   var out = 'blue'
-  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'trackster', 'toastr' ];
+  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'trackster'];
 
   var _ = grunt.util._; 
   var fmt = _.sprintf; 
https://bitbucket.org/galaxy/galaxy-central/commits/5463295a3981/
Changeset:   5463295a3981
User:        martenson
Date:        2013-12-16 00:26:37
Summary:     Merged galaxy/galaxy-central into default
Affected #:  139 files
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/app.py
--- a/lib/galaxy/app.py
+++ b/lib/galaxy/app.py
@@ -1,16 +1,10 @@
 from __future__ import absolute_import
-import sys, os, atexit
+import sys
+import os
 
-from galaxy import config, jobs, util, tools, web
-import galaxy.tools.search
-import galaxy.tools.data
-import tool_shed.galaxy_install
-import tool_shed.tool_shed_registry
-from galaxy.web import security
+from galaxy import config, jobs
 import galaxy.model
-import galaxy.datatypes.registry
 import galaxy.security
-from galaxy.objectstore import build_object_store_from_config
 import galaxy.quota
 from galaxy.tags.tag_handler import GalaxyTagHandler
 from galaxy.visualization.genomes import Genomes
@@ -27,7 +21,8 @@
 import logging
 log = logging.getLogger( __name__ )
 
-class UniverseApplication( object ):
+
+class UniverseApplication( object, config.ConfiguresGalaxyMixin ):
     """Encapsulates the state of a Universe application"""
     def __init__( self, **kwargs ):
         print >> sys.stderr, "python path is: " + ", ".join( sys.path )
@@ -38,92 +33,38 @@
         self.config.check()
         config.configure_logging( self.config )
         self.configure_fluent_log()
-        # Determine the database url
-        if self.config.database_connection:
-            db_url = self.config.database_connection
-        else:
-            db_url = "sqlite:///%s?isolation_level=IMMEDIATE" % self.config.database
-        install_db_url = self.config.install_database_connection
-        # TODO: Consider more aggressive check here that this is not the same
-        # database file under the hood.
-        combined_install_database = not( install_db_url and install_db_url != db_url )
-        # Set up the tool sheds registry
-        if os.path.isfile( self.config.tool_sheds_config ):
-            self.tool_shed_registry = tool_shed.tool_shed_registry.Registry( self.config.root, self.config.tool_sheds_config )
-        else:
-            self.tool_shed_registry = None
-        # Initialize database / check for appropriate schema version.  # If this
-        # is a new installation, we'll restrict the tool migration messaging.
-        from galaxy.model.migrate.check import create_or_verify_database
-        create_or_verify_database( db_url, kwargs.get( 'global_conf', {} ).get( '__file__', None ), self.config.database_engine_options, app=self )
-        if not combined_install_database:
-            from galaxy.model.tool_shed_install.migrate.check import create_or_verify_database as tsi_create_or_verify_database
-            tsi_create_or_verify_database( install_db_url, self.config.install_database_engine_options, app=self )
 
-        # Alert the Galaxy admin to tools that have been moved from the distribution to the tool shed.
-        from tool_shed.galaxy_install.migrate.check import verify_tools
-        verify_tools( self, db_url, kwargs.get( 'global_conf', {} ).get( '__file__', None ), self.config.database_engine_options )
-        # Object store manager
-        self.object_store = build_object_store_from_config(self.config, fsmon=True)
+        self._configure_tool_shed_registry()
+
+        self._configure_object_store( fsmon=True )
+
         # Setup the database engine and ORM
-        from galaxy.model import mapping
-        self.model = mapping.init( self.config.file_path,
-                                   db_url,
-                                   self.config.database_engine_options,
-                                   map_install_models=combined_install_database,
-                                   database_query_profiling_proxy = self.config.database_query_profiling_proxy,
-                                   object_store = self.object_store,
-                                   trace_logger=self.trace_logger,
-                                   use_pbkdf2=self.config.get_bool( 'use_pbkdf2', True ) )
+        config_file = kwargs.get( 'global_conf', {} ).get( '__file__', None )
+        self._configure_models( check_migrate_databases=True, check_migrate_tools=True, config_file=config_file )
 
-        if combined_install_database:
-            log.info("Install database targetting Galaxy's database configuration.")
-            self.install_model = self.model
-        else:
-            from galaxy.model.tool_shed_install import mapping as install_mapping
-            install_db_url = self.config.install_database_connection
-            log.info("Install database using its own connection %s" % install_db_url)
-            install_db_engine_options = self.config.install_database_engine_options
-            self.install_model = install_mapping.init( install_db_url,
-                                                       install_db_engine_options )
         # Manage installed tool shed repositories.
-        self.installed_repository_manager = tool_shed.galaxy_install.InstalledRepositoryManager( self )
-        # Create an empty datatypes registry.
-        self.datatypes_registry = galaxy.datatypes.registry.Registry()
-        # Load proprietary datatypes defined in datatypes_conf.xml files in all installed tool shed repositories.  We
-        # load proprietary datatypes before datatypes in the distribution because Galaxy's default sniffers include some
-        # generic sniffers (eg text,xml) which catch anything, so it's impossible for proprietary sniffers to be used.
-        # However, if there is a conflict (2 datatypes with the same extension) between a proprietary datatype and a datatype
-        # in the Galaxy distribution, the datatype in the Galaxy distribution will take precedence.  If there is a conflict
-        # between 2 proprietary datatypes, the datatype from the repository that was installed earliest will take precedence.
-        self.installed_repository_manager.load_proprietary_datatypes()
-        # Load the data types in the Galaxy distribution, which are defined in self.config.datatypes_config.
-        self.datatypes_registry.load_datatypes( self.config.root, self.config.datatypes_config )
+        from tool_shed.galaxy_install import installed_repository_manager
+        self.installed_repository_manager = installed_repository_manager.InstalledRepositoryManager( self )
+
+        self._configure_datatypes_registry( self.installed_repository_manager )
         galaxy.model.set_datatypes_registry( self.datatypes_registry )
+
         # Security helper
-        self.security = security.SecurityHelper( id_secret=self.config.id_secret )
+        self._configure_security()
         # Tag handler
         self.tag_handler = GalaxyTagHandler()
         # Genomes
         self.genomes = Genomes( self )
         # Data providers registry.
         self.data_provider_registry = DataProviderRegistry()
-        # Initialize tool data tables using the config defined by self.config.tool_data_table_config_path.
-        self.tool_data_tables = galaxy.tools.data.ToolDataTableManager( tool_data_path=self.config.tool_data_path,
-                                                                        config_filename=self.config.tool_data_table_config_path )
-        # Load additional entries defined by self.config.shed_tool_data_table_config into tool data tables.
-        self.tool_data_tables.load_from_config_file( config_filename=self.config.shed_tool_data_table_config,
-                                                     tool_data_path=self.tool_data_tables.tool_data_path,
-                                                     from_shed_config=False )
+
+        self._configure_tool_data_tables( from_shed_config=False )
+
         # Initialize the job management configuration
         self.job_config = jobs.JobConfiguration(self)
-        # Initialize the tools, making sure the list of tool configs includes the reserved migrated_tools_conf.xml file.
-        tool_configs = self.config.tool_configs
-        if self.config.migrated_tools_config not in tool_configs:
-            tool_configs.append( self.config.migrated_tools_config )
-        self.toolbox = tools.ToolBox( tool_configs, self.config.tool_path, self )
-        # Search support for tools
-        self.toolbox_search = galaxy.tools.search.ToolBoxSearch( self.toolbox )
+
+        self._configure_toolbox()
+
         # Load Data Manager
         self.data_managers = DataManagers( self )
         # If enabled, poll respective tool sheds to see if updates are available for any installed tool shed repositories.
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/config.py
--- a/lib/galaxy/config.py
+++ b/lib/galaxy/config.py
@@ -473,6 +473,7 @@
             rval[ key  ] = value
     return rval
 
+
 def configure_logging( config ):
     """
     Allow some basic logging configuration to be read from ini file.
@@ -513,3 +514,110 @@
         sentry_handler.setLevel( logging.WARN )
         root.addHandler( sentry_handler )
 
+
+class ConfiguresGalaxyMixin:
+    """ Shared code for configuring Galaxy-like app objects.
+    """
+
+    def _configure_toolbox( self ):
+        # Initialize the tools, making sure the list of tool configs includes the reserved migrated_tools_conf.xml file.
+        tool_configs = self.config.tool_configs
+        if self.config.migrated_tools_config not in tool_configs:
+            tool_configs.append( self.config.migrated_tools_config )
+        from galaxy import tools
+        self.toolbox = tools.ToolBox( tool_configs, self.config.tool_path, self )
+        # Search support for tools
+        import galaxy.tools.search
+        self.toolbox_search = galaxy.tools.search.ToolBoxSearch( self.toolbox )
+
+    def _configure_tool_data_tables( self, from_shed_config ):
+        from galaxy.tools.data import ToolDataTableManager
+
+        # Initialize tool data tables using the config defined by self.config.tool_data_table_config_path.
+        self.tool_data_tables = ToolDataTableManager( tool_data_path=self.config.tool_data_path,
+                                                      config_filename=self.config.tool_data_table_config_path )
+        # Load additional entries defined by self.config.shed_tool_data_table_config into tool data tables.
+        self.tool_data_tables.load_from_config_file( config_filename=self.config.shed_tool_data_table_config,
+                                                     tool_data_path=self.tool_data_tables.tool_data_path,
+                                                     from_shed_config=from_shed_config )
+
+    def _configure_datatypes_registry( self, installed_repository_manager=None ):
+        from galaxy.datatypes import registry
+        # Create an empty datatypes registry.
+        self.datatypes_registry = registry.Registry()
+        if installed_repository_manager:
+            # Load proprietary datatypes defined in datatypes_conf.xml files in all installed tool shed repositories.  We
+            # load proprietary datatypes before datatypes in the distribution because Galaxy's default sniffers include some
+            # generic sniffers (eg text,xml) which catch anything, so it's impossible for proprietary sniffers to be used.
+            # However, if there is a conflict (2 datatypes with the same extension) between a proprietary datatype and a datatype
+            # in the Galaxy distribution, the datatype in the Galaxy distribution will take precedence.  If there is a conflict
+            # between 2 proprietary datatypes, the datatype from the repository that was installed earliest will take precedence.
+            installed_repository_manager.load_proprietary_datatypes()
+        # Load the data types in the Galaxy distribution, which are defined in self.config.datatypes_config.
+        self.datatypes_registry.load_datatypes( self.config.root, self.config.datatypes_config )
+
+    def _configure_object_store( self, **kwds ):
+        from galaxy.objectstore import build_object_store_from_config
+        self.object_store = build_object_store_from_config( self.config, **kwds )
+
+    def _configure_security( self ):
+        from galaxy.web import security
+        self.security = security.SecurityHelper( id_secret=self.config.id_secret )
+
+    def _configure_tool_shed_registry( self ):
+        import tool_shed.tool_shed_registry
+
+        # Set up the tool sheds registry
+        if os.path.isfile( self.config.tool_sheds_config ):
+            self.tool_shed_registry = tool_shed.tool_shed_registry.Registry( self.config.root, self.config.tool_sheds_config )
+        else:
+            self.tool_shed_registry = None
+
+    def _configure_models( self, check_migrate_databases=False, check_migrate_tools=False, config_file=None ):
+        """
+        Preconditions: object_store must be set on self.
+        """
+        if self.config.database_connection:
+            db_url = self.config.database_connection
+        else:
+            db_url = "sqlite:///%s?isolation_level=IMMEDIATE" % self.config.database
+        install_db_url = self.config.install_database_connection
+        # TODO: Consider more aggressive check here that this is not the same
+        # database file under the hood.
+        combined_install_database = not( install_db_url and install_db_url != db_url )
+        install_db_url = install_db_url or db_url
+
+        if check_migrate_databases:
+            # Initialize database / check for appropriate schema version.  # If this
+            # is a new installation, we'll restrict the tool migration messaging.
+            from galaxy.model.migrate.check import create_or_verify_database
+            create_or_verify_database( db_url, config_file, self.config.database_engine_options, app=self )
+            if not combined_install_database:
+                from galaxy.model.tool_shed_install.migrate.check import create_or_verify_database as tsi_create_or_verify_database
+                tsi_create_or_verify_database( install_db_url, self.config.install_database_engine_options, app=self )
+
+        if check_migrate_tools:
+            # Alert the Galaxy admin to tools that have been moved from the distribution to the tool shed.
+            from tool_shed.galaxy_install.migrate.check import verify_tools
+            verify_tools( self, install_db_url, config_file, self.config.database_engine_options )
+
+        from galaxy.model import mapping
+        self.model = mapping.init( self.config.file_path,
+                                   db_url,
+                                   self.config.database_engine_options,
+                                   map_install_models=combined_install_database,
+                                   database_query_profiling_proxy=self.config.database_query_profiling_proxy,
+                                   object_store=self.object_store,
+                                   trace_logger=getattr(self, "trace_logger", None),
+                                   use_pbkdf2=self.config.get_bool( 'use_pbkdf2', True ) )
+
+        if combined_install_database:
+            log.info("Install database targetting Galaxy's database configuration.")
+            self.install_model = self.model
+        else:
+            from galaxy.model.tool_shed_install import mapping as install_mapping
+            install_db_url = self.config.install_database_connection
+            log.info("Install database using its own connection %s" % install_db_url)
+            install_db_engine_options = self.config.install_database_engine_options
+            self.install_model = install_mapping.init( install_db_url,
+                                                       install_db_engine_options )
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/datatypes/registry.py
--- a/lib/galaxy/datatypes/registry.py
+++ b/lib/galaxy/datatypes/registry.py
@@ -598,6 +598,9 @@
         tool_xml_text = """
             <tool id="__SET_METADATA__" name="Set External Metadata" version="1.0.1" tool_type="set_metadata"><type class="SetMetadataTool" module="galaxy.tools"/>
+              <requirements>
+                  <requirement type="package">samtools</requirement>
+              </requirements><action module="galaxy.tools.actions.metadata" class="SetMetadataToolAction"/><command>$__SET_EXTERNAL_METADATA_COMMAND_LINE__</command><inputs>
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/jobs/__init__.py
--- a/lib/galaxy/jobs/__init__.py
+++ b/lib/galaxy/jobs/__init__.py
@@ -688,10 +688,7 @@
         if self.command_line and self.command_line.startswith( 'python' ):
             self.galaxy_lib_dir = os.path.abspath( "lib" ) # cwd = galaxy root
         # Shell fragment to inject dependencies
-        if self.app.config.use_tool_dependencies:
-            self.dependency_shell_commands = self.tool.build_dependency_shell_commands()
-        else:
-            self.dependency_shell_commands = None
+        self.dependency_shell_commands = self.tool.build_dependency_shell_commands()
         # We need command_line persisted to the db in order for Galaxy to re-queue the job
         # if the server was stopped and restarted before the job finished
         job.command_line = self.command_line
@@ -1435,10 +1432,7 @@
         if self.command_line and self.command_line.startswith( 'python' ):
             self.galaxy_lib_dir = os.path.abspath( "lib" ) # cwd = galaxy root
         # Shell fragment to inject dependencies
-        if self.app.config.use_tool_dependencies:
-            self.dependency_shell_commands = self.tool.build_dependency_shell_commands()
-        else:
-            self.dependency_shell_commands = None
+        self.dependency_shell_commands = self.tool.build_dependency_shell_commands()
         # We need command_line persisted to the db in order for Galaxy to re-queue the job
         # if the server was stopped and restarted before the job finished
         task.command_line = self.command_line
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/jobs/actions/post.py
--- a/lib/galaxy/jobs/actions/post.py
+++ b/lib/galaxy/jobs/actions/post.py
@@ -12,7 +12,7 @@
         form = """
             if (pja.action_type == "%s"){
                 p_str = "<div class='pjaForm toolForm'><span class='action_tag' style='display:none'>"+ pja.action_type + pja.output_name + "</span><div class='toolFormTitle'> %s <br/> on " + pja.output_name + "\
-                <div style='float: right;' class='buttons'><img src='/static/images/delete_icon.png'></div></div><div class='toolFormBody'>";
+                <div style='float: right;' class='buttons'><img src='/static/images/history-buttons/delete_icon.png'></div></div><div class='toolFormBody'>";
                 %s
                 p_str += "</div><div class='toolParamHelp'>%s</div></div>";
             }""" % (action_type, title, content, help)
@@ -20,7 +20,7 @@
         form =  """
             if (pja.action_type == "%s"){
                 p_str = "<div class='pjaForm toolForm'><span class='action_tag' style='display:none'>"+ pja.action_type + "</span><div class='toolFormTitle'> %s \
-                <div style='float: right;' class='buttons'><img src='/static/images/delete_icon.png'></div></div><div class='toolFormBody'>";
+                <div style='float: right;' class='buttons'><img src='/static/images/history-buttons/delete_icon.png'></div></div><div class='toolFormBody'>";
                 %s
                 p_str += "</div><div class='toolParamHelp'>%s</div></div>";
             }""" % (action_type, title, content, help)
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/jobs/runners/__init__.py
--- a/lib/galaxy/jobs/runners/__init__.py
+++ b/lib/galaxy/jobs/runners/__init__.py
@@ -198,7 +198,7 @@
                                 log.exception( "from_work_dir specified a location not in the working directory: %s, %s" % ( source_file, job_wrapper.working_directory ) )
         return output_pairs
 
-    def _handle_metadata_externally(self, job_wrapper):
+    def _handle_metadata_externally( self, job_wrapper, resolve_requirements=False ):
         """
         Set metadata externally. Used by the local and lwr job runners where this
         shouldn't be attached to command-line to execute.
@@ -212,6 +212,12 @@
                                                                             tmp_dir=job_wrapper.working_directory,
                                                                             #we don't want to overwrite metadata that was copied over in init_meta(), as per established behavior
                                                                             kwds={ 'overwrite' : False } )
+            if resolve_requirements:
+                dependency_shell_commands = self.app.datatypes_registry.set_external_metadata_tool.build_dependency_shell_commands()
+                if dependency_shell_commands:
+                    if isinstance( dependency_shell_commands, list ):
+                        dependency_shell_commands = "&&".join( dependency_shell_commands )
+                    external_metadata_script = "%s&&%s" % ( dependency_shell_commands, external_metadata_script )
             log.debug( 'executing external set_meta script for job %d: %s' % ( job_wrapper.job_id, external_metadata_script ) )
             external_metadata_proc = subprocess.Popen( args=external_metadata_script,
                                                        shell=True,
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/jobs/runners/local.py
--- a/lib/galaxy/jobs/runners/local.py
+++ b/lib/galaxy/jobs/runners/local.py
@@ -110,7 +110,7 @@
             job_wrapper.fail( "failure running job", exception=True )
             log.exception("failure running job %d" % job_wrapper.job_id)
             return
-        self._handle_metadata_externally( job_wrapper )
+        self._handle_metadata_externally( job_wrapper, resolve_requirements=True )
         # Finish the job!
         try:
             job_wrapper.finish( stdout, stderr, exit_code )
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/jobs/runners/lwr.py
--- a/lib/galaxy/jobs/runners/lwr.py
+++ b/lib/galaxy/jobs/runners/lwr.py
@@ -204,7 +204,7 @@
             log.exception("failure running job %d" % job_wrapper.job_id)
             return
         if not LwrJobRunner.__remote_metadata( client ):
-            self._handle_metadata_externally( job_wrapper )
+            self._handle_metadata_externally( job_wrapper, resolve_requirements=True )
         # Finish the job
         try:
             job_wrapper.finish( stdout, stderr )
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/model/tool_shed_install/__init__.py
--- a/lib/galaxy/model/tool_shed_install/__init__.py
+++ b/lib/galaxy/model/tool_shed_install/__init__.py
@@ -1,11 +1,11 @@
+import logging
 import os
-
 from galaxy.model.item_attrs import Dictifiable
-
 from galaxy.util.bunch import Bunch
 from galaxy.util import asbool
+from tool_shed.util import common_util
 
-from tool_shed.util import common_util
+log = logging.getLogger( __name__ )
 
 
 class ToolShedRepository( object ):
@@ -61,9 +61,53 @@
     def can_reset_metadata( self ):
         return self.status == self.installation_status.INSTALLED
 
-    @property
-    def can_uninstall( self ):
-        return self.status != self.installation_status.UNINSTALLED
+    def can_uninstall( self, app ):
+        # An installed repository cannot be uninstalled if other installed repositories or installed repository
+        # contents (i.e., tool dependencies) require it.
+        if self.status == self.installation_status.UNINSTALLED:
+            return False
+        this_repository_tup = ( str( self.tool_shed ),
+                                str( self.name ),
+                                str( self.owner ),
+                                str( self.installed_changeset_revision ) )
+        irm = app.installed_repository_manager
+        # See if this repository's current dependencies are restricted to a single circular relationship.  This
+        # means that this repository has a single repository dependency which itself depends upon this repository.
+        # The repository dependency may have other repository dependencies, but that is not relevant here.
+        single_repository_dependency_tup = None
+        installed_repository_dependency_tups = \
+            irm.installed_repository_dependencies_of_installed_repositories.get( this_repository_tup, [] )
+        # If this repository defines a circular relationship to another repository, then the list of tuples
+        # defining installed repository dependencies will include itself.
+        if len( installed_repository_dependency_tups ) == 2:
+            if this_repository_tup in installed_repository_dependency_tups:
+                # We have a single circular dependency definition, so get the other repository.
+                for installed_repository_dependency_tup in installed_repository_dependency_tups:
+                    if installed_repository_dependency_tup != this_repository_tup:
+                        single_repository_dependency_tup = installed_repository_dependency_tup
+                        break
+        if single_repository_dependency_tup is not None:
+            installed_repository_dependency_tups = \
+                irm.installed_repository_dependencies_of_installed_repositories.get( this_repository_tup, [] )
+            if this_repository_tup in installed_repository_dependency_tups:
+                # This repository is a dependency of the single repository upon which it depends, so we have
+                # a single circular relationship and this repository can be uninstalled.
+                return True
+        # Find other installed repositories that require this repository.
+        installed_dependent_repositories = \
+                    irm.installed_dependent_repositories_of_installed_repositories.get( this_repository_tup, [] )
+        if installed_dependent_repositories:
+            # This repository cannot be uninstalled because other installed repositories require it.
+            return False
+        # Find installed tool dependencies that require this repository's installed tool dependencies.
+        installed_tool_dependencies = irm.installed_tool_dependencies_of_installed_repositories.get( this_repository_tup, [] )
+        for td_tup in installed_tool_dependencies:
+            installed_dependent_td_tups = irm.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies.get( td_tup, [] )
+            if installed_dependent_td_tups:
+                # This repository cannot be uninstalled because it contains installed tool dependencies that
+                # are required at run time by other installed tool dependencies.
+                return False
+        return True
 
     @property
     def can_deactivate( self ):
@@ -218,7 +262,7 @@
 
     @property
     def installed_tool_dependencies( self ):
-        """Return the repository's tool dependencies that are currently installed."""
+        """Return the repository's tool dependencies that are currently installed, but possibly in an error state."""
         installed_dependencies = []
         for tool_dependency in self.tool_dependencies:
             if tool_dependency.status in [ ToolDependency.installation_status.INSTALLED,
@@ -313,9 +357,9 @@
         required_repositories_missing_or_being_installed = []
         for required_repository in self.repository_dependencies:
             if required_repository.status in [ self.installation_status.ERROR,
-                                              self.installation_status.INSTALLING,
-                                              self.installation_status.NEVER_INSTALLED,
-                                              self.installation_status.UNINSTALLED ]:
+                                               self.installation_status.INSTALLING,
+                                               self.installation_status.NEVER_INSTALLED,
+                                               self.installation_status.UNINSTALLED ]:
                 required_repositories_missing_or_being_installed.append( required_repository )
         return required_repositories_missing_or_being_installed
 
@@ -507,7 +551,9 @@
 
     @property
     def can_uninstall( self ):
-        return self.status in [ self.installation_status.ERROR, self.installation_status.INSTALLED ]
+        # A tool dependency can be uninstalled only if it is currently in an error state.  Only the containing
+        # repository can be uninstalled if a tool dependency is properly installed.
+        return self.status in [ self.installation_status.ERROR ]
 
     @property
     def can_update( self ):
@@ -516,6 +562,13 @@
                                 self.installation_status.ERROR,
                                 self.installation_status.UNINSTALLED ]
 
+    def get_env_shell_file_path( self, app ):
+        installation_directory = self.installation_directory( app )
+        file_path = os.path.join( installation_directory, 'env.sh' )
+        if os.path.exists( file_path ):
+            return file_path
+        return None
+
     @property
     def in_error_state( self ):
         return self.status == self.installation_status.ERROR
@@ -536,6 +589,10 @@
                                  self.tool_shed_repository.name,
                                  self.tool_shed_repository.installed_changeset_revision )
 
+    @property
+    def is_installed( self ):
+        return self.status == self.installation_status.INSTALLED
+
 
 class ToolVersion( object, Dictifiable ):
     dict_element_visible_keys = ( 'id', 'tool_shed_repository' )
@@ -549,23 +606,23 @@
     def get_previous_version( self, app ):
         context = app.install_model.context
         tva = context.query( app.install_model.ToolVersionAssociation ) \
-                        .filter( app.install_model.ToolVersionAssociation.table.c.tool_id == self.id ) \
-                        .first()
+                     .filter( app.install_model.ToolVersionAssociation.table.c.tool_id == self.id ) \
+                     .first()
         if tva:
             return context.query( app.install_model.ToolVersion ) \
-                             .filter( app.install_model.ToolVersion.table.c.id == tva.parent_id ) \
-                             .first()
+                          .filter( app.install_model.ToolVersion.table.c.id == tva.parent_id ) \
+                          .first()
         return None
 
     def get_next_version( self, app ):
         context = app.install_model.context
         tva = context.query( app.install_model.ToolVersionAssociation ) \
-                        .filter( app.install_model.ToolVersionAssociation.table.c.parent_id == self.id ) \
-                        .first()
+                     .filter( app.install_model.ToolVersionAssociation.table.c.parent_id == self.id ) \
+                     .first()
         if tva:
             return context.query( app.install_model.ToolVersion ) \
-                             .filter( app.install_model.ToolVersion.table.c.id == tva.tool_id ) \
-                             .first()
+                          .filter( app.install_model.ToolVersion.table.c.id == tva.tool_id ) \
+                          .first()
         return None
 
     def get_versions( self, app ):
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/model/tool_shed_install/mapping.py
--- a/lib/galaxy/model/tool_shed_install/mapping.py
+++ b/lib/galaxy/model/tool_shed_install/mapping.py
@@ -7,10 +7,8 @@
 from galaxy.model.base import ModelMapping
 from galaxy.model.orm.engine_factory import build_engine
 
-
 metadata = MetaData()
 
-
 install_model.ToolShedRepository.table = Table( "tool_shed_repository", metadata,
     Column( "id", Integer, primary_key=True ),
     Column( "create_time", DateTime, default=now ),
@@ -72,7 +70,6 @@
     Column( "repository_path", TEXT ),
     Column( "version", Integer ) )
 
-
 mapper( install_model.ToolShedRepository, install_model.ToolShedRepository.table,
     properties=dict( tool_versions=relation( install_model.ToolVersion,
                                              primaryjoin=( install_model.ToolShedRepository.table.c.id == install_model.ToolVersion.table.c.tool_shed_repository_id ),
@@ -112,17 +109,13 @@
     """Connect mappings to the database"""
     # Load the appropriate db module
     engine = build_engine( url, engine_options )
-
     # Connect the metadata to the database.
     metadata.bind = engine
-
-    result = ModelMapping([install_model], engine=engine)
-
+    result = ModelMapping( [ install_model ], engine=engine )
     # Create tables if needed
     if create_tables:
         metadata.create_all()
         # metadata.engine.commit()
-
     result.create_tables = create_tables
     #load local galaxy security policy
     return result
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/tools/__init__.py
--- a/lib/galaxy/tools/__init__.py
+++ b/lib/galaxy/tools/__init__.py
@@ -40,7 +40,7 @@
 from galaxy.tools.actions import DefaultToolAction
 from galaxy.tools.actions.data_source import DataSourceToolAction
 from galaxy.tools.actions.data_manager import DataManagerToolAction
-from galaxy.tools.deps import DependencyManager, INDETERMINATE_DEPENDENCY
+from galaxy.tools.deps import build_dependency_manager
 from galaxy.tools.deps.requirements import parse_requirements_from_xml
 from galaxy.tools.parameters import check_param, params_from_strings, params_to_strings
 from galaxy.tools.parameters.basic import (BaseURLToolParameter,
@@ -700,14 +700,7 @@
         return stored.latest_workflow
 
     def init_dependency_manager( self ):
-        if self.app.config.use_tool_dependencies:
-            dependency_manager_kwds = {
-                'default_base_path': self.app.config.tool_dependency_dir,
-                'conf_file': self.app.config.dependency_resolvers_config_file,
-            }
-            self.dependency_manager = DependencyManager( **dependency_manager_kwds )
-        else:
-            self.dependency_manager = None
+        self.dependency_manager = build_dependency_manager( self.app.config )
 
     @property
     def sa_session( self ):
@@ -1082,6 +1075,22 @@
         :returns: galaxy.jobs.JobDestination -- The destination definition and runner parameters.
         """
         return self.app.job_config.get_destination(self.__get_job_tool_configuration(job_params=job_params).destination)
+    
+    def get_panel_section( self ):
+        for key, item in self.app.toolbox.integrated_tool_panel.items():
+            if item:
+                if key.startswith( 'tool_' ):
+                    if item.id == self.id:
+                        return '', ''
+                if key.startswith( 'section_' ):
+                    section_id = item.id or ''
+                    section_name = item.name or ''
+                    for section_key, section_item in item.elems.items():
+                        if section_key.startswith( 'tool_' ):
+                            if section_item:
+                                if section_item.id == self.id:
+                                    return section_id, section_name
+        return None, None
 
     def parse( self, root, guid=None ):
         """
@@ -3017,6 +3026,8 @@
         if io_details:
             tool_dict[ 'inputs' ] = [ input.to_dict( trans ) for input in self.inputs.values() ]
             tool_dict[ 'outputs' ] = [ output.to_dict() for output in self.outputs.values() ]
+            
+        tool_dict[ 'panel_section_id' ], tool_dict[ 'panel_section_name' ] = self.get_panel_section()
 
         return tool_dict
 
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/tools/deps/__init__.py
--- a/lib/galaxy/tools/deps/__init__.py
+++ b/lib/galaxy/tools/deps/__init__.py
@@ -15,6 +15,31 @@
 from galaxy.util.submodules import submodules
 
 
+def build_dependency_manager( config ):
+    if config.use_tool_dependencies:
+        dependency_manager_kwds = {
+            'default_base_path': config.tool_dependency_dir,
+            'conf_file': config.dependency_resolvers_config_file,
+        }
+        dependency_manager = DependencyManager( **dependency_manager_kwds )
+    else:
+        dependency_manager = NullDependencyManager()
+
+    return dependency_manager
+
+
+class NullDependencyManager( object ):
+
+    def uses_tool_shed_dependencies(self):
+        return False
+
+    def dependency_shell_commands( self, requirements, **kwds ):
+        return []
+
+    def find_dep( self, name, version=None, type='package', **kwds ):
+        return INDETERMINATE_DEPENDENCY
+
+
 class DependencyManager( object ):
     """
     A DependencyManager attempts to resolve named and versioned dependencies by
@@ -76,6 +101,7 @@
         return [
             ToolShedPackageDependencyResolver(self),
             GalaxyPackageDependencyResolver(self),
+            GalaxyPackageDependencyResolver(self, versionless=True),
         ]
 
     def __parse_resolver_conf_xml(self, root):
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/webapps/galaxy/controllers/admin_toolshed.py
--- a/lib/galaxy/webapps/galaxy/controllers/admin_toolshed.py
+++ b/lib/galaxy/webapps/galaxy/controllers/admin_toolshed.py
@@ -265,6 +265,9 @@
             if remove_from_disk_checked:
                 tool_shed_repository.status = trans.install_model.ToolShedRepository.installation_status.UNINSTALLED
                 tool_shed_repository.error_message = None
+                # Remove the uninstalled repository and any tool dependencies from the in-memory dictionaries in the
+                # installed_repository_manager.
+                trans.app.installed_repository_manager.handle_repository_uninstall( tool_shed_repository )
             else:
                 tool_shed_repository.status = trans.install_model.ToolShedRepository.installation_status.DEACTIVATED
             trans.install_model.context.add( tool_shed_repository )
@@ -1564,7 +1567,7 @@
         tool_shed_repository = tool_dependencies[ 0 ].tool_shed_repository
         if kwd.get( 'uninstall_tool_dependencies_button', False ):
             errors = False
-            # Filter tool dependencies to only those that are installed.
+            # Filter tool dependencies to only those that are installed but in an error state.
             tool_dependencies_for_uninstallation = []
             for tool_dependency in tool_dependencies:
                 if tool_dependency.can_uninstall:
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/webapps/galaxy/controllers/forms.py
--- a/lib/galaxy/webapps/galaxy/controllers/forms.py
+++ b/lib/galaxy/webapps/galaxy/controllers/forms.py
@@ -65,6 +65,9 @@
     global_actions = [
         grids.GridAction( "Create new form", dict( controller='forms', action='create_form_definition' ) )
     ]
+    
+    def build_initial_query( self, trans, **kwargs ):
+        return trans.sa_session.query( self.model_class ).join (model.FormDefinition, self.model_class.latest_form_id == model.FormDefinition.id)
 
 class Forms( BaseUIController ):
     # Empty TextField
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/webapps/galaxy/controllers/history.py
--- a/lib/galaxy/webapps/galaxy/controllers/history.py
+++ b/lib/galaxy/webapps/galaxy/controllers/history.py
@@ -838,8 +838,8 @@
             # Set imported history to be user's current history.
             trans.set_history( new_history )
             return trans.show_ok_message(
-                message="""History "%s" has been imported. <br>You can <a href="%s">start using this history</a> or %s."""
-                % ( new_history.name, web.url_for( '/' ), referer_message ), use_panels=True )
+                message="""History "%s" has been imported. <br>You can <a href="%s" onclick="parent.window.location='%s';">start using this history</a> or %s."""
+                % ( new_history.name, web.url_for( '/' ), web.url_for( '/' ), referer_message ), use_panels=True )
         elif not user_history or not user_history.datasets or confirm:
             new_history = import_history.copy()
             new_history.name = "imported: " + new_history.name
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/galaxy/webapps/tool_shed/controllers/repository.py
--- a/lib/galaxy/webapps/tool_shed/controllers/repository.py
+++ b/lib/galaxy/webapps/tool_shed/controllers/repository.py
@@ -932,7 +932,7 @@
         repository_type = kwd.get( 'repository_type', rt_util.UNRESTRICTED )
         if kwd.get( 'create_repository_button', False ):
             error = False
-            message = repository_maintenance_util.validate_repository_name( name, trans.user )
+            message = repository_maintenance_util.validate_repository_name( trans.app, name, trans.user )
             if message:
                 error = True
             if not description:
@@ -2085,7 +2085,7 @@
                 repository.long_description = long_description
                 flush_needed = True
             if repository.times_downloaded == 0 and repo_name != repository.name:
-                message = repository_maintenance_util.validate_repository_name( repo_name, user )
+                message = repository_maintenance_util.validate_repository_name( trans.app, repo_name, user )
                 if message:
                     error = True
                 else:
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/galaxy_install/__init__.py
--- a/lib/tool_shed/galaxy_install/__init__.py
+++ b/lib/tool_shed/galaxy_install/__init__.py
@@ -1,54 +0,0 @@
-"""
-Classes encapsulating the management of repositories installed from Galaxy tool sheds.
-"""
-import os, logging
-import tool_shed.util.shed_util_common
-import tool_shed.util.datatype_util
-from galaxy.model.orm import and_
-from tool_shed.util import xml_util
-
-log = logging.getLogger( __name__ )
-
-class InstalledRepositoryManager( object ):
-    def __init__( self, app ):
-        self.app = app
-        self.install_model = self.app.install_model
-        self.context = self.install_model.context
-        self.tool_configs = self.app.config.tool_configs
-        if self.app.config.migrated_tools_config not in self.tool_configs:
-            self.tool_configs.append( self.app.config.migrated_tools_config )
-        self.installed_repository_dicts = []
-    def get_repository_install_dir( self, tool_shed_repository ):
-        for tool_config in self.tool_configs:
-            tree, error_message = xml_util.parse_xml( tool_config )
-            if tree is None:
-                return None
-            root = tree.getroot()
-            tool_path = root.get( 'tool_path', None )
-            if tool_path:
-                ts = tool_shed.util.shed_util_common.clean_tool_shed_url( tool_shed_repository.tool_shed )
-                relative_path = os.path.join( tool_path,
-                                              ts,
-                                              'repos',
-                                              tool_shed_repository.owner,
-                                              tool_shed_repository.name,
-                                              tool_shed_repository.installed_changeset_revision )
-                if os.path.exists( relative_path ):
-                    return relative_path
-        return None
-    def load_proprietary_datatypes( self ):
-        for tool_shed_repository in self.context.query( self.install_model.ToolShedRepository ) \
-                                                   .filter( and_( self.install_model.ToolShedRepository.table.c.includes_datatypes==True,
-                                                                  self.install_model.ToolShedRepository.table.c.deleted==False ) ) \
-                                                   .order_by( self.install_model.ToolShedRepository.table.c.id ):
-            relative_install_dir = self.get_repository_install_dir( tool_shed_repository )
-            if relative_install_dir:
-                installed_repository_dict = tool_shed.util.datatype_util.load_installed_datatypes( self.app, tool_shed_repository, relative_install_dir )
-                if installed_repository_dict:
-                    self.installed_repository_dicts.append( installed_repository_dict )
-    def load_proprietary_converters_and_display_applications( self, deactivate=False ):
-        for installed_repository_dict in self.installed_repository_dicts:
-            if installed_repository_dict[ 'converter_path' ]:
-                tool_shed.util.datatype_util.load_installed_datatype_converters( self.app, installed_repository_dict, deactivate=deactivate )
-            if installed_repository_dict[ 'display_path' ]:
-                tool_shed.util.datatype_util.load_installed_display_applications( self.app, installed_repository_dict, deactivate=deactivate )
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/galaxy_install/grids/admin_toolshed_grids.py
--- a/lib/tool_shed/galaxy_install/grids/admin_toolshed_grids.py
+++ b/lib/tool_shed/galaxy_install/grids/admin_toolshed_grids.py
@@ -31,6 +31,13 @@
         latest_revision_tip_str = ''
     return '<img src="%s/june_2007_style/blue/ok_small.png" %s/>' % ( url_for( '/static' ), latest_revision_tip_str )
 
+def generate_repository_can_be_uninstalled_img_str( include_mouse_over=False ):
+    if include_mouse_over:
+        can_be_uninstalled_tip_str = 'class="icon-button" title="This repository can be uninstalled"'
+    else:
+        can_be_uninstalled_tip_str = ''
+    return '<img src="%s/images/fugue/toggle-bw.png" %s/>' % ( url_for( '/static' ), can_be_uninstalled_tip_str )
+
 def generate_revision_updates_img_str( include_mouse_over=False ):
     if include_mouse_over:
         revision_updates_tip_str = 'class="icon-button" title="Updates are available in the Tool Shed for this revision"'
@@ -73,6 +80,8 @@
                     tool_shed_status_str += generate_includes_workflows_img_str( include_mouse_over=True )
             else:
                 tool_shed_status_str = generate_unknown_img_str( include_mouse_over=True )
+            if tool_shed_repository.can_uninstall( trans.app ):
+                tool_shed_status_str += generate_repository_can_be_uninstalled_img_str( include_mouse_over=True )
             return tool_shed_status_str
 
 
@@ -210,6 +219,7 @@
         legend_str += '%s  This repository is deprecated in the Tool Shed<br/>' % generate_deprecated_repository_img_str()
         legend_str += '%s  This repository contains exported workflows<br/>' % generate_includes_workflows_img_str()
         legend_str += '%s  Unable to get information from the Tool Shed<br/>' % generate_unknown_img_str()
+        legend_str += '%s  This repository can be uninstalled<br/>' % generate_repository_can_be_uninstalled_img_str()
         return legend_str
 
 
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/galaxy_install/installed_repository_manager.py
--- /dev/null
+++ b/lib/tool_shed/galaxy_install/installed_repository_manager.py
@@ -0,0 +1,368 @@
+"""
+Class encapsulating the management of repositories installed from Galaxy tool sheds.
+"""
+import logging
+import os
+import tool_shed.util.shed_util_common as suc
+from tool_shed.util import datatype_util
+from tool_shed.util import repository_dependency_util
+from tool_shed.util import tool_dependency_util
+from tool_shed.util import xml_util
+from galaxy.model.orm import and_
+
+log = logging.getLogger( __name__ )
+
+
+class InstalledRepositoryManager( object ):
+
+    def __init__( self, app ):
+        """
+        Among other things, keep in in-memory sets of tuples defining installed repositories and tool dependencies along with
+        the relationships between each of them.  This will allow for quick discovery of those repositories or components that
+        can be uninstalled.  The feature allowing a Galaxy administrator to uninstall a repository should not be available to
+        repositories or tool dependency packages that are required by other repositories or their contents (packages). The
+        uninstall feature should be available only at the repository hierarchy level where every dependency will be uninstalled.
+        The exception for this is if an item (repository or tool dependency package) is not in an INSTALLED state - in these
+        cases, the specific item can be uninstalled in order to attempt re-installation.
+        """
+        self.app = app
+        self.install_model = self.app.install_model
+        self.context = self.install_model.context
+        self.tool_configs = self.app.config.tool_configs
+        if self.app.config.migrated_tools_config not in self.tool_configs:
+            self.tool_configs.append( self.app.config.migrated_tools_config )
+        self.installed_repository_dicts = []
+        # Keep an in-memory dictionary whose keys are tuples defining tool_shed_repository objects (whose status is 'Installed')
+        # and whose values are a list of tuples defining tool_shed_repository objects (whose status can be anything) required by
+        # the key.  The value defines the entire repository dependency tree.
+        self.repository_dependencies_of_installed_repositories = {}
+        # Keep an in-memory dictionary whose keys are tuples defining tool_shed_repository objects (whose status is 'Installed')
+        # and whose values are a list of tuples defining tool_shed_repository objects (whose status is 'Installed') required by
+        # the key.  The value defines the entire repository dependency tree.
+        self.installed_repository_dependencies_of_installed_repositories = {}
+        # Keep an in-memory dictionary whose keys are tuples defining tool_shed_repository objects (whose status is 'Installed')
+        # and whose values are a list of tuples defining tool_shed_repository objects (whose status is 'Installed') that require
+        # the key.
+        self.installed_dependent_repositories_of_installed_repositories = {}
+        # Keep an in-memory dictionary whose keys are tuples defining tool_shed_repository objects (whose status is 'Installed')
+        # and whose values are a list of tuples defining its immediate tool_dependency objects (whose status can be anything).
+        # The value defines only the immediate tool dependencies of the repository and does not include any dependencies of the
+        # tool dependencies.
+        self.tool_dependencies_of_installed_repositories = {}
+        # Keep an in-memory dictionary whose keys are tuples defining tool_shed_repository objects (whose status is 'Installed')
+        # and whose values are a list of tuples defining its immediate tool_dependency objects (whose status is 'Installed').
+        # The value defines only the immediate tool dependencies of the repository and does not include any dependencies of the
+        # tool dependencies.
+        self.installed_tool_dependencies_of_installed_repositories = {}
+        # Keep an in-memory dictionary whose keys are tuples defining tool_dependency objects (whose status is 'Installed') and
+        # whose values are a list of tuples defining tool_dependency objects (whose status can be anything) required by the
+        # installed tool dependency at runtime.  The value defines the entire tool dependency tree.
+        self.runtime_tool_dependencies_of_installed_tool_dependencies = {}
+        # Keep an in-memory dictionary whose keys are tuples defining tool_dependency objects (whose status is 'Installed') and
+        # whose values are a list of tuples defining tool_dependency objects (whose status is 'Installed') that require the key
+        # at runtime.  The value defines the entire tool dependency tree.
+        self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies = {}
+        # Load defined dependency relationships for installed tool shed repositories and their contents.
+        self.load_dependency_relationships()
+
+    def add_entry_to_installed_repository_dependencies_of_installed_repositories( self, repository ):
+        """
+        Add an entry to self.installed_repository_dependencies_of_installed_repositories.  A side-effect of this method
+        is the population of self.installed_dependent_repositories_of_installed_repositories.  Since this method discovers
+        all repositories required by the received repository, it can use the list to add entries to the reverse dictionary.
+        """
+        repository_tup = repository_dependency_util.get_repository_tuple_for_installed_repository_manager( repository )
+        tool_shed, name, owner, installed_changeset_revision = repository_tup
+        # Get the list of repository dependencies for this repository.
+        status = self.install_model.ToolShedRepository.installation_status.INSTALLED
+        repository_dependency_tups = \
+            repository_dependency_util.get_repository_dependency_tups_for_repository( self.app, repository, status=status )
+        # Add an entry to self.installed_repository_dependencies_of_installed_repositories.
+        if repository_tup not in self.installed_repository_dependencies_of_installed_repositories:
+            debug_msg = "Adding an entry for revision %s of repository %s owned by %s " % ( installed_changeset_revision, name, owner )
+            debug_msg += "to installed_repository_dependencies_of_installed_repositories."
+            log.debug( debug_msg )
+            self.installed_repository_dependencies_of_installed_repositories[ repository_tup ] = repository_dependency_tups
+        # Use the repository_dependency_tups to add entries to the reverse dictionary
+        # self.installed_dependent_repositories_of_installed_repositories.
+        for required_repository_tup in repository_dependency_tups:
+            debug_msg = "Appending revision %s of repository %s owned by %s " % ( installed_changeset_revision, name, owner )
+            debug_msg += "to all dependent repositories in installed_dependent_repositories_of_installed_repositories."
+            log.debug( debug_msg )
+            if required_repository_tup in self.installed_dependent_repositories_of_installed_repositories:
+                self.installed_dependent_repositories_of_installed_repositories[ required_repository_tup ].append( repository_tup )
+            else:
+                self.installed_dependent_repositories_of_installed_repositories[ required_repository_tup ] = [ repository_tup ]
+
+    def add_entry_to_installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies( self, tool_dependency ):
+        """Add an entry to self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies."""
+        tool_dependency_tup = tool_dependency_util.get_tool_dependency_tuple_for_installed_repository_manager( tool_dependency )
+        if tool_dependency_tup not in self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies:
+            tool_shed_repository_id, name, version, type = tool_dependency_tup
+            debug_msg = "Adding an entry for version %s of %s %s " % ( version, type, name )
+            debug_msg += "to installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies."
+            log.debug( debug_msg )
+            status = self.install_model.ToolDependency.installation_status.INSTALLED
+            installed_runtime_dependent_tool_dependency_tups = \
+                tool_dependency_util.get_runtime_dependent_tool_dependency_tuples( self.app, tool_dependency, status=status )
+            self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies[ tool_dependency_tup ] = \
+                installed_runtime_dependent_tool_dependency_tups
+
+    def add_entry_to_installed_tool_dependencies_of_installed_repositories( self, repository ):
+        """Add an entry to self.installed_tool_dependencies_of_installed_repositories."""
+        repository_tup = repository_dependency_util.get_repository_tuple_for_installed_repository_manager( repository )
+        if repository_tup not in self.installed_tool_dependencies_of_installed_repositories:
+            tool_shed, name, owner, installed_changeset_revision = repository_tup
+            debug_msg = "Adding an entry for revision %s of repository %s owned by %s " % ( installed_changeset_revision, name, owner )
+            debug_msg += "to installed_tool_dependencies_of_installed_repositories."
+            log.debug( debug_msg )
+            installed_tool_dependency_tups = []
+            for tool_dependency in repository.tool_dependencies:
+                if tool_dependency.status == self.app.install_model.ToolDependency.installation_status.INSTALLED:
+                    tool_dependency_tup = tool_dependency_util.get_tool_dependency_tuple_for_installed_repository_manager( tool_dependency )
+                    installed_tool_dependency_tups.append( tool_dependency_tup )
+            self.installed_tool_dependencies_of_installed_repositories[ repository_tup ] = installed_tool_dependency_tups
+
+    def add_entry_to_repository_dependencies_of_installed_repositories( self, repository ):
+        """Add an entry to self.repository_dependencies_of_installed_repositories."""
+        repository_tup = repository_dependency_util.get_repository_tuple_for_installed_repository_manager( repository )
+        if repository_tup not in self.repository_dependencies_of_installed_repositories:
+            tool_shed, name, owner, installed_changeset_revision = repository_tup
+            debug_msg = "Adding an entry for revision %s of repository %s owned by %s " % ( installed_changeset_revision, name, owner )
+            debug_msg += "to repository_dependencies_of_installed_repositories."
+            log.debug( debug_msg )
+            repository_dependency_tups = \
+                repository_dependency_util.get_repository_dependency_tups_for_repository( self.app, repository, status=None )
+            self.repository_dependencies_of_installed_repositories[ repository_tup ] = repository_dependency_tups
+
+    def add_entry_to_runtime_tool_dependencies_of_installed_tool_dependencies( self, tool_dependency ):
+        """Add an entry to self.runtime_tool_dependencies_of_installed_tool_dependencies."""
+        tool_dependency_tup = tool_dependency_util.get_tool_dependency_tuple_for_installed_repository_manager( tool_dependency )
+        if tool_dependency_tup not in self.runtime_tool_dependencies_of_installed_tool_dependencies:
+            tool_shed_repository_id, name, version, type = tool_dependency_tup
+            debug_msg = "Adding an entry for version %s of %s %s " % ( version, type, name )
+            debug_msg += "to runtime_tool_dependencies_of_installed_tool_dependencies."
+            log.debug( debug_msg )
+            runtime_dependent_tool_dependency_tups = \
+                tool_dependency_util.get_runtime_dependent_tool_dependency_tuples( self.app, tool_dependency, status=None )
+            self.runtime_tool_dependencies_of_installed_tool_dependencies[ tool_dependency_tup ] = \
+                runtime_dependent_tool_dependency_tups
+
+    def add_entry_to_tool_dependencies_of_installed_repositories( self, repository ):
+        """Add an entry to self.tool_dependencies_of_installed_repositories."""
+        repository_tup = repository_dependency_util.get_repository_tuple_for_installed_repository_manager( repository )
+        if repository_tup not in self.tool_dependencies_of_installed_repositories:
+            tool_shed, name, owner, installed_changeset_revision = repository_tup
+            debug_msg = "Adding an entry for revision %s of repository %s owned by %s " % ( installed_changeset_revision, name, owner )
+            debug_msg += "to tool_dependencies_of_installed_repositories."
+            log.debug( debug_msg )
+            tool_dependency_tups = []
+            for tool_dependency in repository.tool_dependencies:
+                tool_dependency_tup = tool_dependency_util.get_tool_dependency_tuple_for_installed_repository_manager( tool_dependency )
+                tool_dependency_tups.append( tool_dependency_tup )
+            self.tool_dependencies_of_installed_repositories[ repository_tup ] = tool_dependency_tups
+
+    def get_containing_repository_for_tool_dependency( self, tool_dependency_tup ):
+        tool_shed_repository_id, name, version, type = tool_dependency_tup
+        return self.app.install_model.context.query( self.app.install_model.ToolShedRepository ).get( tool_shed_repository_id )
+            
+    def get_repository_install_dir( self, tool_shed_repository ):
+        for tool_config in self.tool_configs:
+            tree, error_message = xml_util.parse_xml( tool_config )
+            if tree is None:
+                return None
+            root = tree.getroot()
+            tool_path = root.get( 'tool_path', None )
+            if tool_path:
+                ts = suc.clean_tool_shed_url( tool_shed_repository.tool_shed )
+                relative_path = os.path.join( tool_path,
+                                              ts,
+                                              'repos',
+                                              str( tool_shed_repository.owner ),
+                                              str( tool_shed_repository.name ),
+                                              str( tool_shed_repository.installed_changeset_revision ) )
+                if os.path.exists( relative_path ):
+                    return relative_path
+        return None
+
+    def handle_repository_install( self, repository ):
+        """Load the dependency relationships for a repository that was just installed or reinstalled."""
+        # Populate self.repository_dependencies_of_installed_repositories.
+        self.add_entry_to_repository_dependencies_of_installed_repositories( repository )
+        # Populate self.installed_repository_dependencies_of_installed_repositories.
+        self.add_entry_to_installed_repository_dependencies_of_installed_repositories( repository )
+        # Populate self.tool_dependencies_of_installed_repositories.
+        self.add_entry_to_tool_dependencies_of_installed_repositories( repository )
+        # Populate self.installed_tool_dependencies_of_installed_repositories.
+        self.add_entry_to_installed_tool_dependencies_of_installed_repositories( repository )
+        for tool_dependency in repository.tool_dependencies:
+            # Populate self.runtime_tool_dependencies_of_installed_tool_dependencies.
+            self.add_entry_to_runtime_tool_dependencies_of_installed_tool_dependencies( tool_dependency )
+            # Populate self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies.
+            self.add_entry_to_installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies( tool_dependency )
+
+    def handle_repository_uninstall( self, repository ):
+        """Remove the dependency relationships for a repository that was just uninstalled."""
+        for tool_dependency in repository.tool_dependencies:
+            tool_dependency_tup = tool_dependency_util.get_tool_dependency_tuple_for_installed_repository_manager( tool_dependency )
+            # Remove this tool_dependency from all values in
+            # self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies
+            altered_installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies = {}
+            for td_tup, installed_runtime_dependent_tool_dependency_tups in \
+                self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies.items():
+                if tool_dependency_tup in installed_runtime_dependent_tool_dependency_tups:
+                    # Remove the tool_dependency from the list.
+                    installed_runtime_dependent_tool_dependency_tups.remove( tool_dependency_tup )
+                # Add the possibly altered list to the altered dictionary.
+                altered_installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies[ td_tup ] = \
+                    installed_runtime_dependent_tool_dependency_tups
+            self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies = \
+                altered_installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies
+            # Remove the entry for this tool_dependency from self.runtime_tool_dependencies_of_installed_tool_dependencies.
+            self.remove_entry_from_runtime_tool_dependencies_of_installed_tool_dependencies( tool_dependency )
+            # Remove the entry for this tool_dependency from 
+            # self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies.
+            self.remove_entry_from_installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies( tool_dependency )
+        # Remove this repository's entry from self.installed_tool_dependencies_of_installed_repositories.
+        self.remove_entry_from_installed_tool_dependencies_of_installed_repositories( repository )
+        # Remove this repository's entry from self.tool_dependencies_of_installed_repositories
+        self.remove_entry_from_tool_dependencies_of_installed_repositories( repository )
+        # Remove this repository's entry from self.installed_repository_dependencies_of_installed_repositories.
+        self.remove_entry_from_installed_repository_dependencies_of_installed_repositories( repository )
+        # Remove this repository's entry from self.repository_dependencies_of_installed_repositories.
+        self.remove_entry_from_repository_dependencies_of_installed_repositories( repository )
+
+    def handle_tool_dependency_install( self, repository, tool_dependency ):
+        """Load the dependency relationships for a tool dependency that was just installed independently of its containing repository."""
+        # The received repository must have a status of 'Installed'.  The value of tool_dependency.status will either be
+        # 'Installed' or 'Error', but we only need to change the in-memory dictionaries if it is 'Installed'.
+        if tool_dependency.is_installed:
+            # Populate self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies.
+            self.add_entry_to_installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies( tool_dependency )
+            # Populate self.installed_tool_dependencies_of_installed_repositories.
+            repository_tup = repository_dependency_util.get_repository_tuple_for_installed_repository_manager( repository )
+            tool_dependency_tup = tool_dependency_util.get_tool_dependency_tuple_for_installed_repository_manager( tool_dependency )
+            if repository_tup in self.installed_tool_dependencies_of_installed_repositories:
+                self.installed_tool_dependencies_of_installed_repositories[ repository_tup ].append( tool_dependency_tup )
+            else:
+                self.installed_tool_dependencies_of_installed_repositories[ repository_tup ] = [ tool_dependency_tup ]
+
+    def load_dependency_relationships( self ):
+        """Load relationships for all installed repositories and tool dependencies into in-memnory dictionaries."""
+        # Get the list of installed tool shed repositories.
+        for repository in self.context.query( self.app.install_model.ToolShedRepository ) \
+                                      .filter( self.app.install_model.ToolShedRepository.table.c.status ==
+                                               self.app.install_model.ToolShedRepository.installation_status.INSTALLED ):
+            # Populate self.repository_dependencies_of_installed_repositories.
+            self.add_entry_to_repository_dependencies_of_installed_repositories( repository )
+            # Populate self.installed_repository_dependencies_of_installed_repositories.
+            self.add_entry_to_installed_repository_dependencies_of_installed_repositories( repository )
+            # Populate self.tool_dependencies_of_installed_repositories.
+            self.add_entry_to_tool_dependencies_of_installed_repositories( repository )
+            # Populate self.installed_tool_dependencies_of_installed_repositories.
+            self.add_entry_to_installed_tool_dependencies_of_installed_repositories( repository )
+        # Get the list of installed tool dependencies.
+        for tool_dependency in self.context.query( self.app.install_model.ToolDependency ) \
+                                           .filter( self.app.install_model.ToolDependency.table.c.status ==
+                                                    self.app.install_model.ToolDependency.installation_status.INSTALLED ):
+            # Populate self.runtime_tool_dependencies_of_installed_tool_dependencies.
+            self.add_entry_to_runtime_tool_dependencies_of_installed_tool_dependencies( tool_dependency )
+            # Populate self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies.
+            self.add_entry_to_installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies( tool_dependency )
+
+    def load_proprietary_datatypes( self ):
+        for tool_shed_repository in self.context.query( self.install_model.ToolShedRepository ) \
+                                                   .filter( and_( self.install_model.ToolShedRepository.table.c.includes_datatypes==True,
+                                                                  self.install_model.ToolShedRepository.table.c.deleted==False ) ) \
+                                                   .order_by( self.install_model.ToolShedRepository.table.c.id ):
+            relative_install_dir = self.get_repository_install_dir( tool_shed_repository )
+            if relative_install_dir:
+                installed_repository_dict = datatype_util.load_installed_datatypes( self.app, tool_shed_repository, relative_install_dir )
+                if installed_repository_dict:
+                    self.installed_repository_dicts.append( installed_repository_dict )
+
+    def load_proprietary_converters_and_display_applications( self, deactivate=False ):
+        for installed_repository_dict in self.installed_repository_dicts:
+            if installed_repository_dict[ 'converter_path' ]:
+                datatype_util.load_installed_datatype_converters( self.app, installed_repository_dict, deactivate=deactivate )
+            if installed_repository_dict[ 'display_path' ]:
+                datatype_util.load_installed_display_applications( self.app, installed_repository_dict, deactivate=deactivate )
+
+    def remove_entry_from_installed_repository_dependencies_of_installed_repositories( self, repository ):
+        """
+        Remove an entry from self.installed_repository_dependencies_of_installed_repositories.  A side-effect of this method
+        is removal of appropriate value items from self.installed_dependent_repositories_of_installed_repositories.
+        """
+        # Remove tuples defining this repository from value lists in self.installed_dependent_repositories_of_installed_repositories.
+        repository_tup = repository_dependency_util.get_repository_tuple_for_installed_repository_manager( repository )
+        tool_shed, name, owner, installed_changeset_revision = repository_tup
+        altered_installed_dependent_repositories_of_installed_repositories = {}
+        for r_tup, v_tups in self.installed_dependent_repositories_of_installed_repositories.items():
+            if repository_tup in v_tups:
+                debug_msg = "Removing entry for revision %s of repository %s owned by %s " % \
+                    ( installed_changeset_revision, name, owner )
+                r_tool_shed, r_name, r_owner, r_installed_changeset_revision = r_tup
+                debug_msg += "from the dependent list for revision %s of repository %s owned by %s " % \
+                    ( r_installed_changeset_revision, r_name, r_owner )
+                debug_msg += "in installed_repository_dependencies_of_installed_repositories."
+                log.debug( debug_msg )                
+                v_tups.remove( repository_tup )
+            altered_installed_dependent_repositories_of_installed_repositories[ r_tup ] = v_tups
+        self.installed_dependent_repositories_of_installed_repositories = \
+            altered_installed_dependent_repositories_of_installed_repositories
+        # Remove this repository's entry from self.installed_repository_dependencies_of_installed_repositories.
+        if repository_tup in self.installed_repository_dependencies_of_installed_repositories:
+            debug_msg = "Removing entry for revision %s of repository %s owned by %s " % ( installed_changeset_revision, name, owner )
+            debug_msg += "from installed_repository_dependencies_of_installed_repositories."
+            log.debug( debug_msg )
+            del self.installed_repository_dependencies_of_installed_repositories[ repository_tup ]
+
+    def remove_entry_from_installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies( self, tool_dependency ):
+        """Remove an entry from self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies."""
+        tool_dependency_tup = tool_dependency_util.get_tool_dependency_tuple_for_installed_repository_manager( tool_dependency )
+        if tool_dependency_tup in self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies:
+            tool_shed_repository_id, name, version, type = tool_dependency_tup
+            debug_msg = "Removing entry for version %s of %s %s " % ( version, type, name )
+            debug_msg += "from installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies."
+            log.debug( debug_msg )
+            del self.installed_runtime_dependent_tool_dependencies_of_installed_tool_dependencies[ tool_dependency_tup ]
+
+    def remove_entry_from_installed_tool_dependencies_of_installed_repositories( self, repository ):
+        """Remove an entry from self.installed_tool_dependencies_of_installed_repositories."""
+        repository_tup = repository_dependency_util.get_repository_tuple_for_installed_repository_manager( repository )
+        if repository_tup in self.installed_tool_dependencies_of_installed_repositories:
+            tool_shed, name, owner, installed_changeset_revision = repository_tup
+            debug_msg = "Removing entry for revision %s of repository %s owned by %s " % ( installed_changeset_revision, name, owner )
+            debug_msg += "from installed_tool_dependencies_of_installed_repositories."
+            log.debug( debug_msg )
+            del self.installed_tool_dependencies_of_installed_repositories[ repository_tup ]
+
+    def remove_entry_from_repository_dependencies_of_installed_repositories( self, repository ):
+        """Remove an entry from self.repository_dependencies_of_installed_repositories."""
+        repository_tup = repository_dependency_util.get_repository_tuple_for_installed_repository_manager( repository )
+        if repository_tup in self.repository_dependencies_of_installed_repositories:
+            tool_shed, name, owner, installed_changeset_revision = repository_tup
+            debug_msg = "Removing entry for revision %s of repository %s owned by %s " % ( installed_changeset_revision, name, owner )
+            debug_msg += "from repository_dependencies_of_installed_repositories."
+            log.debug( debug_msg )
+            del self.repository_dependencies_of_installed_repositories[ repository_tup ]
+
+    def remove_entry_from_runtime_tool_dependencies_of_installed_tool_dependencies( self, tool_dependency ):
+        """Remove an entry from self.runtime_tool_dependencies_of_installed_tool_dependencies."""
+        tool_dependency_tup = tool_dependency_util.get_tool_dependency_tuple_for_installed_repository_manager( tool_dependency )
+        if tool_dependency_tup in self.runtime_tool_dependencies_of_installed_tool_dependencies:
+            tool_shed_repository_id, name, version, type = tool_dependency_tup
+            debug_msg = "Removing entry for version %s of %s %s from runtime_tool_dependencies_of_installed_tool_dependencies." % \
+                ( version, type, name )
+            log.debug( debug_msg )
+            del self.runtime_tool_dependencies_of_installed_tool_dependencies[ tool_dependency_tup ]
+
+    def remove_entry_from_tool_dependencies_of_installed_repositories( self, repository ):
+        """Remove an entry from self.tool_dependencies_of_installed_repositories."""
+        repository_tup = repository_dependency_util.get_repository_tuple_for_installed_repository_manager( repository )
+        if repository_tup in self.tool_dependencies_of_installed_repositories:
+            tool_shed, name, owner, installed_changeset_revision = repository_tup
+            debug_msg = "Removing entry for revision %s of repository %s owned by %s from tool_dependencies_of_installed_repositories." % \
+                ( installed_changeset_revision, name, owner )
+            log.debug( debug_msg )
+            del self.tool_dependencies_of_installed_repositories[ repository_tup ]
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/galaxy_install/migrate/common.py
--- a/lib/tool_shed/galaxy_install/migrate/common.py
+++ b/lib/tool_shed/galaxy_install/migrate/common.py
@@ -2,18 +2,10 @@
 import os
 import sys
 import galaxy.config
-import galaxy.datatypes.registry
-from galaxy import tools
-from galaxy.tools.data import ToolDataTableManager
-from galaxy.web import security
-import galaxy.model.mapping
-import galaxy.tools.search
-from galaxy.objectstore import build_object_store_from_config
-import tool_shed.tool_shed_registry
 from tool_shed.galaxy_install import install_manager
 
 
-class MigrateToolsApplication( object ):
+class MigrateToolsApplication( object, galaxy.config.ConfiguresGalaxyMixin ):
     """Encapsulates the state of a basic Galaxy Universe application in order to initiate the Install Manager"""
 
     def __init__( self, tools_migration_config ):
@@ -33,41 +25,23 @@
         for key, value in config_parser.items( "app:main" ):
             galaxy_config_dict[ key ] = value
         self.config = galaxy.config.Configuration( **galaxy_config_dict )
-        if not self.config.database_connection:
-            self.config.database_connection = "sqlite:///%s?isolation_level=IMMEDIATE" % self.config.database
+
         self.config.update_integrated_tool_panel = True
-        self.object_store = build_object_store_from_config( self.config )
-        # Security helper
-        self.security = security.SecurityHelper( id_secret=self.config.id_secret )
-        # Setup the database engine and ORM
-        self.model = galaxy.model.mapping.init( self.config.file_path,
-                                                self.config.database_connection,
-                                                engine_options={},
-                                                create_tables=False,
-                                                object_store=self.object_store )
-        # Create an empty datatypes registry.
-        self.datatypes_registry = galaxy.datatypes.registry.Registry()
-        # Load the data types in the Galaxy distribution, which are defined in self.config.datatypes_config.
-        self.datatypes_registry.load_datatypes( self.config.root, self.config.datatypes_config )
-        # Initialize tool data tables using the config defined by self.config.tool_data_table_config_path.
-        self.tool_data_tables = ToolDataTableManager( tool_data_path=self.config.tool_data_path,
-                                                      config_filename=self.config.tool_data_table_config_path )
-        # Load additional entries defined by self.config.shed_tool_data_table_config into tool data tables.
-        self.tool_data_tables.load_from_config_file( config_filename=self.config.shed_tool_data_table_config,
-                                                     tool_data_path=self.tool_data_tables.tool_data_path,
-                                                     from_shed_config=True )
-        # Initialize the tools, making sure the list of tool configs includes the reserved migrated_tools_conf.xml file.
-        tool_configs = self.config.tool_configs
-        if self.config.migrated_tools_config not in tool_configs:
-            tool_configs.append( self.config.migrated_tools_config )
-        self.toolbox = tools.ToolBox( tool_configs, self.config.tool_path, self )
-        # Search support for tools
-        self.toolbox_search = galaxy.tools.search.ToolBoxSearch( self.toolbox )
-        # Set up the tool sheds registry.
-        if os.path.isfile( self.config.tool_sheds_config ):
-            self.tool_shed_registry = tool_shed.tool_shed_registry.Registry( self.config.root, self.config.tool_sheds_config )
-        else:
-            self.tool_shed_registry = None
+
+        self._configure_object_store()
+
+        self._configure_security()
+
+        self._configure_models()
+
+        self._configure_datatypes_registry( )
+
+        self._configure_tool_data_tables( from_shed_config=True )
+
+        self._configure_toolbox()
+
+        self._configure_tool_shed_registry()
+
         # Get the latest tool migration script number to send to the Install manager.
         latest_migration_script_number = int( tools_migration_config.split( '_' )[ 0 ] )
         # The value of migrated_tools_config is migrated_tools_conf.xml, and is reserved for containing only those tools that have been
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/galaxy_install/repository_util.py
--- a/lib/tool_shed/galaxy_install/repository_util.py
+++ b/lib/tool_shed/galaxy_install/repository_util.py
@@ -504,8 +504,8 @@
     query = trans.install_model.context.query( trans.install_model.ToolShedRepository ).filter( or_( *clause_list ) )
     return encoded_kwd, query, tool_shed_repositories, encoded_repository_ids
 
-def install_tool_shed_repository( trans, tool_shed_repository, repo_info_dict, tool_panel_section_key, shed_tool_conf, tool_path, install_tool_dependencies,
-                                  reinstalling=False ):
+def install_tool_shed_repository( trans, tool_shed_repository, repo_info_dict, tool_panel_section_key, shed_tool_conf, tool_path,
+                                  install_tool_dependencies, reinstalling=False ):
     if tool_panel_section_key:
         try:
             tool_section = trans.app.toolbox.tool_panel[ tool_panel_section_key ]
@@ -579,7 +579,11 @@
                                                                                         tool_dependencies_config=tool_dependencies_config,
                                                                                         tool_dependencies=tool_shed_repository.tool_dependencies )
             suc.remove_dir( work_dir )
-        suc.update_tool_shed_repository_status( trans.app, tool_shed_repository, trans.install_model.ToolShedRepository.installation_status.INSTALLED )
+        suc.update_tool_shed_repository_status( trans.app,
+                                                tool_shed_repository,
+                                                trans.install_model.ToolShedRepository.installation_status.INSTALLED )
+        # Add the installed repository and any tool dependencies to the in-memory dictionaries in the installed_repository_manager.
+        trans.app.installed_repository_manager.handle_repository_install( tool_shed_repository )
     else:
         # An error occurred while cloning the repository, so reset everything necessary to enable another attempt.
         set_repository_attributes( trans,
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/galaxy_install/update_manager.py
--- a/lib/tool_shed/galaxy_install/update_manager.py
+++ b/lib/tool_shed/galaxy_install/update_manager.py
@@ -30,7 +30,7 @@
             # repository.  This information includes items like newer installable repository revisions, current revision updates, whether
             # the repository revision is the latest installable revision, and whether the repository has been deprecated in the tool shed.
             for repository in self.context.query( self.app.install_model.ToolShedRepository ) \
-                                             .filter( self.app.install_model.ToolShedRepository.table.c.deleted == False ):
+                                          .filter( self.app.install_model.ToolShedRepository.table.c.deleted == False ):
                 tool_shed_status_dict = suc.get_tool_shed_status_for_installed_repository( self.app, repository )
                 if tool_shed_status_dict:
                     if tool_shed_status_dict != repository.tool_shed_status:
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/scripts/check_repositories_for_functional_tests.py
--- a/lib/tool_shed/scripts/check_repositories_for_functional_tests.py
+++ b/lib/tool_shed/scripts/check_repositories_for_functional_tests.py
@@ -253,6 +253,11 @@
                         # a test_environment entry.  If we use it we need to temporarily eliminate it from the list of tool_test_results_dicts
                         # since it will be re-inserted later.
                         tool_test_results_dict = tool_test_results_dicts.pop( 0 )
+                    elif len( tool_test_results_dict ) == 2 and \
+                        'test_environment' in tool_test_results_dict and 'missing_test_components' in tool_test_results_dict:
+                        # We can re-use tool_test_results_dict if its only entries are "test_environment" and "missing_test_components".
+                        # In this case, some tools are missing tests components while others are not.
+                        tool_test_results_dict = tool_test_results_dicts.pop( 0 )
                     else:
                         # The latest tool_test_results_dict has been populated with the results of a test run, so it cannot be used.
                         tool_test_results_dict = {}
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/scripts/check_tool_dependency_definition_repositories.py
--- a/lib/tool_shed/scripts/check_tool_dependency_definition_repositories.py
+++ b/lib/tool_shed/scripts/check_tool_dependency_definition_repositories.py
@@ -41,6 +41,7 @@
 
 class RepositoriesApplication( object ):
     """Encapsulates the state of a Universe application"""
+
     def __init__( self, config ):
         if config.database_connection is False:
             config.database_connection = "sqlite:///%s?isolation_level=IMMEDIATE" % config.database
@@ -133,11 +134,12 @@
     # Do not check metadata records that have an entry in the skip_tool_tests table, since they won't be tested anyway.
     skip_metadata_ids = select( [ app.model.SkipToolTest.table.c.repository_metadata_id ] )
     # Get the list of metadata records to check, restricting it to records that have not been flagged do_not_test.
-    for repository_metadata in app.sa_session.query( app.model.RepositoryMetadata ) \
-                                         .filter( and_( app.model.RepositoryMetadata.table.c.downloadable == True,
-                                                        app.model.RepositoryMetadata.table.c.do_not_test == False,
-                                                        app.model.RepositoryMetadata.table.c.repository_id.in_( tool_dependency_defintion_repository_ids ),
-                                                        not_( app.model.RepositoryMetadata.table.c.id.in_( skip_metadata_ids ) ) ) ):
+    for repository_metadata in \
+        app.sa_session.query( app.model.RepositoryMetadata ) \
+                      .filter( and_( app.model.RepositoryMetadata.table.c.downloadable == True,
+                                     app.model.RepositoryMetadata.table.c.do_not_test == False,
+                                     app.model.RepositoryMetadata.table.c.repository_id.in_( tool_dependency_defintion_repository_ids ),
+                                     not_( app.model.RepositoryMetadata.table.c.id.in_( skip_metadata_ids ) ) ) ):
         records_checked += 1
         # Check the next repository revision.
         changeset_revision = str( repository_metadata.changeset_revision )
@@ -161,9 +163,31 @@
                 invalid_metadata += 1
             if not info_only:
                 # Create the tool_test_results_dict dictionary, using the dictionary from the previous test run if available.
-                if repository_metadata.tool_test_results:
-                    tool_test_results_dict = repository_metadata.tool_test_results
+                if repository_metadata.tool_test_results is not None:
+                    # We'll listify the column value in case it uses the old approach of storing the results of only a single test run.
+                    tool_test_results_dicts = listify( repository_metadata.tool_test_results )
                 else:
+                    tool_test_results_dicts = []
+                if tool_test_results_dicts:
+                    # Inspect the tool_test_results_dict for the last test run in case it contains only a test_environment
+                    # entry.  This will occur with multiple runs of this script without running the associated
+                    # install_and_test_tool_sed_repositories.sh script which will further populate the tool_test_results_dict.
+                    tool_test_results_dict = tool_test_results_dicts[ 0 ]
+                    if len( tool_test_results_dict ) <= 1:
+                        # We can re-use the mostly empty tool_test_results_dict for this run because it is either empty or it contains only
+                        # a test_environment entry.  If we use it we need to temporarily eliminate it from the list of tool_test_results_dicts
+                        # since it will be re-inserted later.
+                        tool_test_results_dict = tool_test_results_dicts.pop( 0 )
+                    elif len( tool_test_results_dict ) == 2 and \
+                        'test_environment' in tool_test_results_dict and 'missing_test_components' in tool_test_results_dict:
+                        # We can re-use tool_test_results_dict if its only entries are "test_environment" and "missing_test_components".
+                        # In this case, some tools are missing tests components while others are not.
+                        tool_test_results_dict = tool_test_results_dicts.pop( 0 )
+                    else:
+                        # The latest tool_test_results_dict has been populated with the results of a test run, so it cannot be used.
+                        tool_test_results_dict = {}
+                else:
+                    # Create a new dictionary for the most recent test run.
                     tool_test_results_dict = {}
                 # Initialize the tool_test_results_dict dictionary with the information about the current test environment.
                 test_environment_dict = tool_test_results_dict.get( 'test_environment', {} )
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/util/common_install_util.py
--- a/lib/tool_shed/util/common_install_util.py
+++ b/lib/tool_shed/util/common_install_util.py
@@ -71,8 +71,8 @@
 
 def get_dependencies_for_repository( trans, tool_shed_url, repo_info_dict, includes_tool_dependencies ):
     """
-    Return dictionaries containing the sets of installed and missing tool dependencies and repository dependencies associated with the repository defined
-    by the received repo_info_dict.
+    Return dictionaries containing the sets of installed and missing tool dependencies and repository dependencies associated
+    with the repository defined by the received repo_info_dict.
     """
     repository = None
     installed_rd = {}
@@ -102,7 +102,8 @@
         # Discover all repository dependencies and retrieve information for installing them.
         all_repo_info_dict = get_required_repo_info_dicts( trans, tool_shed_url, util.listify( repo_info_dict ) )
         has_repository_dependencies = all_repo_info_dict.get( 'has_repository_dependencies', False )
-        has_repository_dependencies_only_if_compiling_contained_td = all_repo_info_dict.get( 'has_repository_dependencies_only_if_compiling_contained_td', False )
+        has_repository_dependencies_only_if_compiling_contained_td = \
+            all_repo_info_dict.get( 'has_repository_dependencies_only_if_compiling_contained_td', False )
         includes_tools_for_display_in_tool_panel = all_repo_info_dict.get( 'includes_tools_for_display_in_tool_panel', False )
         includes_tool_dependencies = all_repo_info_dict.get( 'includes_tool_dependencies', False )
         includes_tools = all_repo_info_dict.get( 'includes_tools', False )
@@ -138,7 +139,8 @@
         # We have a single repository with (possibly) no defined repository dependencies.
         all_repo_info_dict = get_required_repo_info_dicts( trans, tool_shed_url, util.listify( repo_info_dict ) )
         has_repository_dependencies = all_repo_info_dict.get( 'has_repository_dependencies', False )
-        has_repository_dependencies_only_if_compiling_contained_td = all_repo_info_dict.get( 'has_repository_dependencies_only_if_compiling_contained_td', False )
+        has_repository_dependencies_only_if_compiling_contained_td = \
+            all_repo_info_dict.get( 'has_repository_dependencies_only_if_compiling_contained_td', False )
         includes_tools_for_display_in_tool_panel = all_repo_info_dict.get( 'includes_tools_for_display_in_tool_panel', False )
         includes_tool_dependencies = all_repo_info_dict.get( 'includes_tool_dependencies', False )
         includes_tools = all_repo_info_dict.get( 'includes_tools', False )
@@ -202,8 +204,8 @@
             # Get the description from the metadata in case it has a value.
             repository_dependencies = metadata.get( 'repository_dependencies', {} )
             description = repository_dependencies.get( 'description', None )
-            # We need to add a root_key entry to one or both of installed_repository_dependencies dictionary and the missing_repository_dependencies
-            # dictionaries for proper display parsing.
+            # We need to add a root_key entry to one or both of installed_repository_dependencies dictionary and the
+            # missing_repository_dependencies dictionaries for proper display parsing.
             root_key = container_util.generate_repository_dependencies_key_for_repository( repository.tool_shed,
                                                                                            repository.name,
                                                                                            repository.owner,
@@ -460,7 +462,7 @@
                     if tool_dependency.can_install:
                         # The database record is currently in a state that allows us to install the package on the file system.
                         try:
-                            dependencies_ignored = app.toolbox.dependency_manager and not app.toolbox.dependency_manager.uses_tool_shed_dependencies()
+                            dependencies_ignored = not app.toolbox.dependency_manager.uses_tool_shed_dependencies()
                             if dependencies_ignored:
                                 log.debug( "Skipping package %s because tool shed dependency resolver not enabled." % str( package_name ) )
                                 # Tool dependency resolves have been configured and they do not include the tool shed. Do not install package.
@@ -489,6 +491,8 @@
                         if tool_dependency and tool_dependency.status in [ app.install_model.ToolDependency.installation_status.INSTALLED,
                                                                            app.install_model.ToolDependency.installation_status.ERROR ]:
                             installed_tool_dependencies.append( tool_dependency )
+                            # Add the tool_dependency to the in-memory dictionaries in the installed_repository_manager.
+                            app.installed_repository_manager.handle_tool_dependency_install( tool_shed_repository, tool_dependency )
         elif elem.tag == 'set_environment':
             # <set_environment version="1.0">
             #    <environment_variable name="R_SCRIPT_PATH"action="set_to">$REPOSITORY_INSTALL_DIR</environment_variable>
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/util/container_util.py
--- a/lib/tool_shed/util/container_util.py
+++ b/lib/tool_shed/util/container_util.py
@@ -5,7 +5,6 @@
 from galaxy.web.framework.helpers import time_ago
 from tool_shed.util import common_util
 from tool_shed.util import readme_util
-from tool_shed.util.tool_dependency_util import tool_dependency_is_orphan
 import tool_shed.util.shed_util_common as suc
 
 log = logging.getLogger( __name__ )
@@ -269,7 +268,7 @@
     """Tool dependency object"""
 
     def __init__( self, id=None, name=None, version=None, type=None, readme=None, installation_status=None, repository_id=None,
-                  tool_dependency_id=None, is_orphan=None ):
+                  tool_dependency_id=None ):
         self.id = id
         self.name = name
         self.version = version
@@ -278,15 +277,6 @@
         self.installation_status = installation_status
         self.repository_id = repository_id
         self.tool_dependency_id = tool_dependency_id
-        # The designation of a ToolDependency into the "orphan" category has evolved over time, and is significantly restricted since the
-        # introduction of the TOOL_DEPENDENCY_DEFINITION repository type.  This designation is still critical, however, in that it handles
-        # the case where a repository contains both tools and a tool_dependencies.xml file, but the definition in the tool_dependencies.xml
-        # file is in no way related to anything defined by any of the contained tool's requirements tag sets.  This is important in that it
-        # is often a result of a typo (e.g., dependency name or version) that differs between the tool dependency definition within the
-        # tool_dependencies.xml file and what is defined in the tool config's <requirements> tag sets.  In these cases, the user should be
-        # presented with a warning message, and this warning message is is in fact displayed if the following is_orphan attribute is True.
-        # This is tricky because in some cases it may be intentional, and tool dependencies that are categorized as "orphan" are in fact valid.
-        self.is_orphan = is_orphan
 
     @property
     def listify( self ):
@@ -318,42 +308,6 @@
         self.repository_metadata_id = repository_metadata_id
         self.repository_id = repository_id
 
-def add_orphan_settings_to_tool_dependencies( tool_dependencies, tools ):
-    """Inspect all received tool dependencies and label those that are orphans within the repository."""
-    #orphan_env_dependencies = orphan_tool_dependencies.get( 'set_environment', None )
-    new_tool_dependencies = {}
-    for td_key, requirements_dict in tool_dependencies.items():
-        if td_key in [ 'set_environment' ]:
-            # "set_environment": [{"name": "R_SCRIPT_PATH", "type": "set_environment"}]
-            new_set_environment_dict_list = []
-            for env_requirements_dict in requirements_dict:
-                try:
-                    name = env_requirements_dict[ 'name' ]
-                    type = env_requirements_dict[ 'type' ]
-                    if tool_dependency_is_orphan( type, name, None, tools ):
-                        env_requirements_dict[ 'is_orphan' ] = True
-                except Exception, e:
-                    name = str( e )
-                    type = 'unknown'
-                    is_orphan = 'unknown'
-                new_set_environment_dict_list.append( env_requirements_dict )
-            new_tool_dependencies[ td_key ] = new_set_environment_dict_list
-        else:
-            # {"R/2.15.1": {"name": "R", "readme": "some string", "type": "package", "version": "2.15.1"}
-            try:
-                name = requirements_dict[ 'name' ]
-                type = requirements_dict[ 'type' ]
-                version = requirements_dict[ 'version'] 
-                if tool_dependency_is_orphan( type, name, version, tools ):
-                    requirements_dict[ 'is_orphan' ] = True
-            except Exception, e:
-                name = str( e )
-                type = 'unknown'
-                version = 'unknown'
-                is_orphan = 'unknown'
-            new_tool_dependencies[ td_key ] = requirements_dict
-    return new_tool_dependencies
-
 def build_data_managers_folder( trans, folder_id, data_managers, label=None ):
     """Return a folder hierarchy containing Data Managers."""
     if data_managers:
@@ -849,8 +803,6 @@
                         if 'tools' not in exclude:
                             tools = metadata.get( 'tools', [] )
                             tools.extend( metadata.get( 'invalid_tools', [] ) )
-                            if tools:
-                                tool_dependencies = add_orphan_settings_to_tool_dependencies( tool_dependencies, tools )
                     folder_id, tool_dependencies_root_folder = build_tool_dependencies_folder( trans,
                                                                                                folder_id,
                                                                                                tool_dependencies,
@@ -1055,13 +1007,6 @@
             tool_dependency_id += 1
             if dependency_key in [ 'set_environment' ]:
                 for set_environment_dict in requirements_dict:
-                    if trans.webapp.name == 'tool_shed':
-                        is_orphan = set_environment_dict.get( 'is_orphan', False )
-                    else:
-                        # This is probably not necessary to display in Galaxy.
-                        is_orphan = False
-                    if is_orphan:
-                        folder.description = not_used_by_local_tools_description
                     try:
                         name = set_environment_dict.get( 'name', None )
                         type = set_environment_dict[ 'type' ]
@@ -1086,16 +1031,9 @@
                                                       readme=None,
                                                       installation_status=installation_status,
                                                       repository_id=repository_id,
-                                                      tool_dependency_id=td_id,
-                                                      is_orphan=is_orphan )
+                                                      tool_dependency_id=td_id )
                     folder.tool_dependencies.append( tool_dependency )
             else:
-                if trans.webapp.name == 'tool_shed':
-                    is_orphan = requirements_dict.get( 'is_orphan', False )
-                else:
-                    is_orphan = False
-                if is_orphan:
-                    folder.description = not_used_by_local_tools_description
                 try:
                     name = requirements_dict[ 'name' ]
                     version = requirements_dict[ 'version' ]
@@ -1122,8 +1060,7 @@
                                                   readme=None,
                                                   installation_status=installation_status,
                                                   repository_id=repository_id,
-                                                  tool_dependency_id=td_id,
-                                                  is_orphan=is_orphan )
+                                                  tool_dependency_id=td_id )
                 folder.tool_dependencies.append( tool_dependency )
     else:
         tool_dependencies_root_folder = None
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/util/metadata_util.py
--- a/lib/tool_shed/util/metadata_util.py
+++ b/lib/tool_shed/util/metadata_util.py
@@ -940,7 +940,12 @@
             for required_file in ttb.required_files:
                 value, extra = required_file
                 required_files.append( ( value ) )
-            inputs = [ input for input in ttb.inputs ]
+            inputs = []
+            for input_field, values in ttb.inputs.iteritems():
+                if len( values ) == 1:
+                    inputs.append( ( input_field, values[0] ) )
+                else:
+                    inputs.append( ( input_field, values ) )
             outputs = []
             for output in ttb.outputs:
                 name, file_name, extra = output
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/util/repository_dependency_util.py
--- a/lib/tool_shed/util/repository_dependency_util.py
+++ b/lib/tool_shed/util/repository_dependency_util.py
@@ -355,14 +355,53 @@
                                                           handled_key_rd_dicts,
                                                           circular_repository_dependencies )
             elif key_rd_dicts_to_be_processed:
-                handle_next_repository_dependency( trans, key_rd_dicts_to_be_processed, all_repository_dependencies, handled_key_rd_dicts, circular_repository_dependencies )
+                handle_next_repository_dependency( trans,
+                                                   key_rd_dicts_to_be_processed,
+                                                   all_repository_dependencies,
+                                                   handled_key_rd_dicts,
+                                                   circular_repository_dependencies )
         elif key_rd_dicts_to_be_processed:
-            handle_next_repository_dependency( trans, key_rd_dicts_to_be_processed, all_repository_dependencies, handled_key_rd_dicts, circular_repository_dependencies )
+            handle_next_repository_dependency( trans,
+                                               key_rd_dicts_to_be_processed,
+                                               all_repository_dependencies,
+                                               handled_key_rd_dicts,
+                                               circular_repository_dependencies )
     elif key_rd_dicts_to_be_processed:
-        handle_next_repository_dependency( trans, key_rd_dicts_to_be_processed, all_repository_dependencies, handled_key_rd_dicts, circular_repository_dependencies )
+        handle_next_repository_dependency( trans,
+                                           key_rd_dicts_to_be_processed,
+                                           all_repository_dependencies,
+                                           handled_key_rd_dicts,
+                                           circular_repository_dependencies )
     all_repository_dependencies = prune_invalid_repository_dependencies( all_repository_dependencies )
     return all_repository_dependencies
 
+def get_repository_dependency_tups_for_repository( app, repository, dependency_tups=None, status=None ):
+    """
+    Return a list of of tuples defining tool_shed_repository objects (whose status can be anything) required by the
+    received repository.  The returned list defines the entire repository dependency tree.
+    """
+    if dependency_tups is None:
+        dependency_tups = []
+    repository_tup = get_repository_tuple_for_installed_repository_manager( repository )
+    for rrda in repository.required_repositories:
+        repository_dependency = rrda.repository_dependency
+        required_repository = repository_dependency.repository
+        if status is None or required_repository.status == status:
+            required_repository_tup = get_repository_tuple_for_installed_repository_manager( required_repository )
+            if required_repository_tup == repository_tup:
+                # We have a circular repository dependency relationship, skip this entry.
+                continue
+            if required_repository_tup not in dependency_tups:
+                dependency_tups.append( required_repository_tup )
+                return get_repository_dependency_tups_for_repository( app, required_repository, dependency_tups=dependency_tups )
+    return dependency_tups
+
+def get_repository_tuple_for_installed_repository_manager( repository ):
+    return ( str( repository.tool_shed ),
+             str( repository.name ),
+             str( repository.owner ),
+             str( repository.installed_changeset_revision ) )
+
 def get_updated_changeset_revisions_for_repository_dependencies( trans, key_rd_dicts ):
     updated_key_rd_dicts = []
     for key_rd_dict in key_rd_dicts:
@@ -373,9 +412,10 @@
         if suc.tool_shed_is_this_tool_shed( rd_toolshed ):
             repository = suc.get_repository_by_name_and_owner( trans.app, rd_name, rd_owner )
             if repository:
-                repository_metadata = metadata_util.get_repository_metadata_by_repository_id_changeset_revision( trans,
-                                                                                                                 trans.security.encode_id( repository.id ),
-                                                                                                                 rd_changeset_revision )
+                repository_metadata = \
+                    metadata_util.get_repository_metadata_by_repository_id_changeset_revision( trans,
+                                                                                               trans.security.encode_id( repository.id ),
+                                                                                               rd_changeset_revision )
                 if repository_metadata:
                     # The repository changeset_revision is installable, so no updates are available.
                     new_key_rd_dict = {}
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/util/repository_maintenance_util.py
--- a/lib/tool_shed/util/repository_maintenance_util.py
+++ b/lib/tool_shed/util/repository_maintenance_util.py
@@ -137,7 +137,7 @@
                         results_message += 'This Tool Shed does not have the category <b>%s</b> so it will not be associated with this repository.' % \
                             str( category_name )
                     else:
-                       category_ids.append( trans.security.encode_id( category.id ) )
+                        category_ids.append( trans.security.encode_id( category.id ) )
                 # Create the repository record in the database.
                 repository, create_message = create_repository( trans,
                                                                 name,
@@ -157,15 +157,18 @@
             import_results_tups.append( ( ( str( name ), str( username ) ), results_message ) )
     return import_results_tups
 
-def validate_repository_name( name, user ):
+def validate_repository_name( app, name, user ):
     # Repository names must be unique for each user, must be at least four characters
     # in length and must contain only lower-case letters, numbers, and the '_' character.
     if name in [ 'None', None, '' ]:
         return 'Enter the required repository name.'
     if name in [ 'repos' ]:
         return "The term <b>%s</b> is a reserved word in the tool shed, so it cannot be used as a repository name." % name
-    for repository in user.active_repositories:
-        if repository.name == name:
+    check_existing = suc.get_repository_by_name_and_owner( app, name, user.username )
+    if check_existing is not None:
+        if check_existing.deleted:
+            return 'You have a deleted repository named <b>%s</b>, so choose a different name.' % name
+        else:
             return "You already have a repository named <b>%s</b>, so choose a different name." % name
     if len( name ) < 4:
         return "Repository names must be at least 4 characters in length."
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a lib/tool_shed/util/tool_dependency_util.py
--- a/lib/tool_shed/util/tool_dependency_util.py
+++ b/lib/tool_shed/util/tool_dependency_util.py
@@ -4,6 +4,7 @@
 from galaxy import eggs
 from galaxy import util
 from galaxy.model.orm import and_
+from galaxy.model.orm import or_
 import tool_shed.util.shed_util_common as suc
 import tool_shed.repository_types.util as rt_util
 from tool_shed.util import xml_util
@@ -13,9 +14,10 @@
 
 def add_installation_directories_to_tool_dependencies( trans, tool_dependencies ):
     """
-    Determine the path to the installation directory for each of the received tool dependencies.  This path will be displayed within the tool dependencies
-    container on the select_tool_panel_section or reselect_tool_panel_section pages when installing or reinstalling repositories that contain tools with the
-    defined tool dependencies.  The list of tool dependencies may be associated with more than a single repository.
+    Determine the path to the installation directory for each of the received tool dependencies.  This path will be
+    displayed within the tool dependencies container on the select_tool_panel_section or reselect_tool_panel_section
+    pages when installing or reinstalling repositories that contain tools with the defined tool dependencies.  The
+    list of tool dependencies may be associated with more than a single repository.
     """
     for dependency_key, requirements_dict in tool_dependencies.items():
         if dependency_key in [ 'set_environment' ]:
@@ -64,7 +66,10 @@
     return tool_dependency
 
 def create_tool_dependency_objects( app, tool_shed_repository, relative_install_dir, set_status=True ):
-    """Create or update a ToolDependency for each entry in tool_dependencies_config.  This method is called when installing a new tool_shed_repository."""
+    """
+    Create or update a ToolDependency for each entry in tool_dependencies_config.  This method is called when
+    installing a new tool_shed_repository.
+    """
     tool_dependency_objects = []
     shed_config_dict = tool_shed_repository.get_shed_config_dict( app )
     if shed_config_dict.get( 'tool_path' ):
@@ -82,12 +87,13 @@
             name = elem.get( 'name', None )
             version = elem.get( 'version', None )
             if name and version:
+                status = app.install_model.ToolDependency.installation_status.NEVER_INSTALLED
                 tool_dependency = create_or_update_tool_dependency( app,
                                                                     tool_shed_repository,
                                                                     name=name,
                                                                     version=version,
                                                                     type=tool_dependency_type,
-                                                                    status=app.install_model.ToolDependency.installation_status.NEVER_INSTALLED,
+                                                                    status=status,
                                                                     set_status=set_status )
                 tool_dependency_objects.append( tool_dependency )
         elif tool_dependency_type == 'set_environment':
@@ -96,12 +102,13 @@
                 name = env_elem.get( 'name', None )
                 action = env_elem.get( 'action', None )
                 if name and action:
+                    status = app.install_model.ToolDependency.installation_status.NEVER_INSTALLED
                     tool_dependency = create_or_update_tool_dependency( app,
                                                                         tool_shed_repository,
                                                                         name=name,
                                                                         version=None,
                                                                         type=tool_dependency_type,
-                                                                        status=app.install_model.ToolDependency.installation_status.NEVER_INSTALLED,
+                                                                        status=status,
                                                                         set_status=set_status )
                     tool_dependency_objects.append( tool_dependency )
     return tool_dependency_objects
@@ -185,12 +192,14 @@
 def generate_message_for_repository_type_change( trans, repository ):
     message = ''
     if repository.can_change_type_to( trans.app, rt_util.TOOL_DEPENDENCY_DEFINITION ):
-        tool_dependency_definition_type_class = trans.app.repository_types_registry.get_class_by_label( rt_util.TOOL_DEPENDENCY_DEFINITION )
-        message += "This repository currently contains a single file named <b>%s</b>.  If additional files will " % suc.TOOL_DEPENDENCY_DEFINITION_FILENAME
-        message += "not be added to this repository, then it's type should be set to <b>%s</b>.<br/>" % tool_dependency_definition_type_class.label
+        tool_dependency_definition_type_class = \
+            trans.app.repository_types_registry.get_class_by_label( rt_util.TOOL_DEPENDENCY_DEFINITION )
+        message += "This repository currently contains a single file named <b>%s</b>.  If additional files will " % \
+            suc.TOOL_DEPENDENCY_DEFINITION_FILENAME
+        message += "not be added to this repository, then it's type should be set to <b>%s</b>.<br/>" % \
+            tool_dependency_definition_type_class.label
     return message
-                
-                
+
 def get_download_url_for_platform( url_templates, platform_info_dict ):
     '''
     Compare the dict returned by get_platform_info() with the values specified in the url_template element. Return
@@ -289,6 +298,31 @@
     env_sh_file_path = os.path.join( env_sh_file_dir, 'env.sh' )
     return env_sh_file_path
 
+def get_runtime_dependent_tool_dependency_tuples( app, tool_dependency, status=None ):
+    """
+    Return the list of tool dependency objects that require the received tool dependency at run time.  The returned
+    list will be filtered by the received status if it is not None.  This method is called only from Galaxy.
+    """
+    runtime_dependent_tool_dependency_tups = []
+    required_env_shell_file_path = tool_dependency.get_env_shell_file_path( app )
+    if required_env_shell_file_path:
+        required_env_shell_file_path = os.path.abspath( required_env_shell_file_path )
+    if required_env_shell_file_path is not None:
+        for td in app.install_model.context.query( app.install_model.ToolDependency ):
+            if status is None or td.status == status:
+                env_shell_file_path = td.get_env_shell_file_path( app )
+                if env_shell_file_path is not None:
+                    try:
+                        contents = open( env_shell_file_path, 'r' ).read()
+                    except Exception, e:
+                        contents = None
+                        log.debug( 'Error reading file %s, so cannot determine if package %s requires package %s at run time: %s' % \
+                            ( str( env_shell_file_path ), str( td.name ), str( tool_dependency.name ), str( e ) ) )
+                    if contents is not None and contents.find( required_env_shell_file_path ) >= 0:
+                        td_tuple = get_tool_dependency_tuple_for_installed_repository_manager( td )
+                        runtime_dependent_tool_dependency_tups.append( td_tuple )
+    return runtime_dependent_tool_dependency_tups
+
 def get_tool_dependency( trans, id ):
     """Get a tool_dependency from the database via id"""
     return trans.install_model.context.query( trans.install_model.ToolDependency ).get( trans.security.decode_id( id ) )
@@ -353,6 +387,13 @@
                                               repository_name,
                                               repository_changeset_revision ) )
 
+def get_tool_dependency_tuple_for_installed_repository_manager( tool_dependency ):
+    if tool_dependency.type is None:
+        type = None
+    else:
+        type = str( tool_dependency.type )
+    return ( tool_dependency.tool_shed_repository_id, str( tool_dependency.name ), str( tool_dependency.version ), type )
+
 def handle_tool_dependency_installation_error( app, tool_dependency, error_message, remove_installation_path=False ):
     # Since there was an installation error, remove the installation directory because the install_package method uses 
     # this: "if os.path.exists( install_dir ):". Setting remove_installation_path to True should rarely occur. It is
@@ -479,6 +520,7 @@
     return installed_tool_dependencies, missing_tool_dependencies
 
 def remove_tool_dependency( app, tool_dependency ):
+    """The received tool_dependency must be in an error state."""
     context = app.install_model.context
     dependency_install_dir = tool_dependency.installation_directory( app )
     removed, error_message = remove_tool_dependency_installation_directory( dependency_install_dir )
@@ -487,6 +529,9 @@
         tool_dependency.error_message = None
         context.add( tool_dependency )
         context.flush()
+        # Since the received tool_dependency is in an error state, nothing will need to be changed in any
+        # of the in-memory dictionaries in the installed_repository_manager because changing the state from
+        # error to uninstalled requires no in-memory changes..
     return removed, error_message
 
 def remove_tool_dependency_installation_directory( dependency_install_dir ):
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a static/scripts/galaxy.workflow_editor.canvas.js
--- a/static/scripts/galaxy.workflow_editor.canvas.js
+++ b/static/scripts/galaxy.workflow_editor.canvas.js
@@ -360,7 +360,7 @@
                                 workflow.has_changes = true;
                                 canvas_manager.draw_overview();
                             })))
-                    .tooltip({delay:500, title: "Flag this as a workflow output.  All non-flagged outputs will be hidden." });
+                    .tooltip({delay:500, title: "Mark dataset as a workflow output. All unmarked datasets will be hidden." });
                 callout.css({
                         top: '50%',
                         margin:'-8px 0px 0px 0px',
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a static/scripts/mvc/base-mvc.js
--- a/static/scripts/mvc/base-mvc.js
+++ b/static/scripts/mvc/base-mvc.js
@@ -173,6 +173,7 @@
 
 //==============================================================================
 var HiddenUntilActivatedViewMixin = /** @lends hiddenUntilActivatedMixin# */{
+//TODO: since this is a mixin, consider moving toggle, hidden into HUAVOptions
 
     /** */
     hiddenUntilActivated : function( $activator, options ){
@@ -184,7 +185,9 @@
             showSpeed       : 'fast'
         };
         _.extend( this.HUAVOptions, options || {});
+        /** has this been shown already (and onshowFirstTime called)? */
         this.HUAVOptions.hasBeenShown = this.HUAVOptions.$elementShown.is( ':visible' );
+        this.hidden = this.isHidden();
 
         if( $activator ){
             var mixin = this;
@@ -194,21 +197,36 @@
         }
     },
 
+    isHidden : function(){
+        return ( this.HUAVOptions.$elementShown.is( ':hidden' ) );
+    },
+
     /** */
     toggle : function(){
         // can be called manually as well with normal toggle arguments
-        if( this.HUAVOptions.$elementShown.is( ':hidden' ) ){
+        //TODO: better as a callback (when the show/hide is actually done)
+        // show
+        if( this.hidden ){
             // fire the optional fns on the first/each showing - good for render()
             if( !this.HUAVOptions.hasBeenShown ){
                 if( _.isFunction( this.HUAVOptions.onshowFirstTime ) ){
                     this.HUAVOptions.hasBeenShown = true;
                     this.HUAVOptions.onshowFirstTime.call( this );
                 }
-            } else {
-                if( _.isFunction( this.HUAVOptions.onshow ) ){
-                    this.HUAVOptions.onshow.call( this );
-                }
             }
+            if( _.isFunction( this.HUAVOptions.onshow ) ){
+                this.HUAVOptions.onshow.call( this );
+                this.trigger( 'hiddenUntilActivated:shown', this );
+            }
+            this.hidden = false;
+
+        // hide
+        } else {
+            if( _.isFunction( this.HUAVOptions.onhide ) ){
+                this.HUAVOptions.onhide.call( this );
+                this.trigger( 'hiddenUntilActivated:hidden', this );
+            }
+            this.hidden = true;
         }
         return this.HUAVOptions.showFn.apply( this.HUAVOptions.$elementShown, arguments );
     }
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 5463295a3981c8487effbbbaa127fe028737491a static/scripts/mvc/dataset/hda-base.js
--- a/static/scripts/mvc/dataset/hda-base.js
+++ b/static/scripts/mvc/dataset/hda-base.js
@@ -46,11 +46,11 @@
         /** is the view currently in selection mode? */
         this.selectable = attributes.selectable || false;
         /** is the view currently selected? */
-        this.selected = attributes.selected || false;
-        /** is the body of this hda view expanded/not. */
-        this.expanded = attributes.expanded || false;
-        /** is the body of this hda view expanded/not. */
-        this.draggable = attributes.draggable || false;
+        this.selected   = attributes.selected || false;
+        /** is the body of this hda view expanded/not? */
+        this.expanded   = attributes.expanded || false;
+        /** is the body of this hda view expanded/not? */
+        this.draggable  = attributes.draggable || false;
         
         this._setUpListeners();
     },
@@ -423,7 +423,7 @@
     toggleBodyVisibility : function( event, expand ){
         // bail (with propagation) if keydown and not space or enter
         var KEYCODE_SPACE = 32, KEYCODE_RETURN = 13;
-        if( ( event.type === 'keydown' )
+        if( event && ( event.type === 'keydown' )
         &&  !( event.keyCode === KEYCODE_SPACE || event.keyCode === KEYCODE_RETURN ) ){
             return true;
         }
This diff is so big that we needed to truncate the remainder.
https://bitbucket.org/galaxy/galaxy-central/commits/70aca9eee9d4/
Changeset:   70aca9eee9d4
User:        martenson
Date:        2013-12-16 00:22:01
Summary:     activation link tweak: show proper message when trying to activate already active account
Affected #:  1 file
diff -r 0522f7504094bd5be2be97846e695a0c790bf929 -r 70aca9eee9d442a88e49dc99e08ff3b6bbea73d0 lib/galaxy/webapps/galaxy/controllers/user.py
--- a/lib/galaxy/webapps/galaxy/controllers/user.py
+++ b/lib/galaxy/webapps/galaxy/controllers/user.py
@@ -826,19 +826,22 @@
 
         if email is None or activation_token is None:
             #  We don't have the email or activation_token, show error.
-            return trans.show_error_message( "You are using wrong activation link. Try to log-in and we will send you a new activation email.<br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller="root", action="index" )
+            return trans.show_error_message( "You are using wrong activation link. Try to log-in and we will send you a new activation email. <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller="root", action="index" )
         else:
             #  Find the user
             user = trans.sa_session.query( trans.app.model.User ).filter( trans.app.model.User.table.c.email==email ).first()
+            # If the user is active already don't try to activate            
+            if user.active == True:
+                return trans.show_ok_message( "Your account is already active. Nothing has changed. <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
             if user.activation_token == activation_token:
                 user.activation_token = None
                 user.active = True
                 trans.sa_session.add(user)
                 trans.sa_session.flush()
-                return trans.show_ok_message( "Your account has been successfully activated!<br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
+                return trans.show_ok_message( "Your account has been successfully activated! <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
             else:
                 #  Tokens don't match. Activation is denied.
-                return trans.show_error_message( "You are using wrong activation link. Try to log in and we will send you a new activation email.<br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
+                return trans.show_error_message( "You are using wrong activation link. Try to log in and we will send you a new activation email. <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
         return
 
     def __get_user_type_form_definition( self, trans, user=None, **kwd ):
https://bitbucket.org/galaxy/galaxy-central/commits/d5be36722eb3/
Changeset:   d5be36722eb3
Branch:      data_library
User:        martenson
Date:        2013-12-16 17:14:59
Summary:     Closed branch data_library
Affected #:  0 files
https://bitbucket.org/galaxy/galaxy-central/commits/2b83e7980707/
Changeset:   2b83e7980707
User:        martenson
Date:        2013-12-16 17:42:28
Summary:     closing data_library branch
Affected #:  0 files
https://bitbucket.org/galaxy/galaxy-central/commits/0b2f53ad1a20/
Changeset:   0b2f53ad1a20
User:        martenson
Date:        2013-12-16 18:01:35
Summary:     Merge own default 2nd head
Affected #:  1 file
diff -r 2b83e798070736ea7279d5e4e33194f66f8ef7df -r 0b2f53ad1a2025dbfd7c15a76f329fc5e8a859ef lib/galaxy/webapps/galaxy/controllers/user.py
--- a/lib/galaxy/webapps/galaxy/controllers/user.py
+++ b/lib/galaxy/webapps/galaxy/controllers/user.py
@@ -826,19 +826,22 @@
 
         if email is None or activation_token is None:
             #  We don't have the email or activation_token, show error.
-            return trans.show_error_message( "You are using wrong activation link. Try to log-in and we will send you a new activation email.<br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller="root", action="index" )
+            return trans.show_error_message( "You are using wrong activation link. Try to log-in and we will send you a new activation email. <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller="root", action="index" )
         else:
             #  Find the user
             user = trans.sa_session.query( trans.app.model.User ).filter( trans.app.model.User.table.c.email==email ).first()
+            # If the user is active already don't try to activate            
+            if user.active == True:
+                return trans.show_ok_message( "Your account is already active. Nothing has changed. <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
             if user.activation_token == activation_token:
                 user.activation_token = None
                 user.active = True
                 trans.sa_session.add(user)
                 trans.sa_session.flush()
-                return trans.show_ok_message( "Your account has been successfully activated!<br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
+                return trans.show_ok_message( "Your account has been successfully activated! <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
             else:
                 #  Tokens don't match. Activation is denied.
-                return trans.show_error_message( "You are using wrong activation link. Try to log in and we will send you a new activation email.<br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
+                return trans.show_error_message( "You are using wrong activation link. Try to log in and we will send you a new activation email. <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
         return
 
     def __get_user_type_form_definition( self, trans, user=None, **kwd ):
https://bitbucket.org/galaxy/galaxy-central/commits/16d53da082b8/
Changeset:   16d53da082b8
User:        martenson
Date:        2013-12-17 19:22:55
Summary:     added security checks when downloading lddas in archives
Affected #:  1 file
diff -r 0b2f53ad1a2025dbfd7c15a76f329fc5e8a859ef -r 16d53da082b89db1318a2a688aa8a18470b02f59 lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- a/lib/galaxy/webapps/galaxy/api/lda_datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -13,15 +13,17 @@
 import urllib
 import urllib2
 import zipfile
+from paste.httpexceptions import HTTPBadRequest
+from galaxy.exceptions import ItemAccessibilityException, MessageException, ItemDeletionException
 from galaxy.security import Action
 from galaxy import util, web
 from galaxy.util.streamball import StreamBall
-from galaxy.web.base.controller import BaseAPIController, UsesLibraryMixinItems
+from galaxy.web.base.controller import BaseAPIController, UsesVisualizationMixin
 
 import logging
 log = logging.getLogger( __name__ )
 
-class DatasetsController( BaseAPIController, UsesLibraryMixinItems ):
+class LibraryDatasetsController( BaseAPIController, UsesVisualizationMixin ):
 
     @web.expose_api
     def show( self, trans, id, **kwd ):
@@ -31,7 +33,7 @@
         """
         # Get dataset.
         try:
-            dataset = self.get_library_dataset( trans, id = id )
+            dataset = self.get_library_dataset( trans, id = id, check_ownership=False, check_accessible=True )
         except Exception, e:
             return str( e )
         try:
@@ -65,12 +67,22 @@
             datasets_to_download = util.listify( datasets_to_download )
             for dataset_id in datasets_to_download:
                 try:
-                    ldda = trans.sa_session.query( trans.app.model.LibraryDatasetDatasetAssociation ).get( trans.security.decode_id( dataset_id ) )
-                    assert not ldda.dataset.purged
+                    ldda = self.get_hda_or_ldda( trans, hda_ldda='ldda', dataset_id=dataset_id )
                     lddas.append( ldda )
-                except:
-                    ldda = None
-                    message += "Invalid library dataset id (%s) specified.  " % str( dataset_id )
+                except ItemAccessibilityException:
+                    trans.response.status = 403
+                    return 'Insufficient rights to access library dataset with id: (%s)'  % str( dataset_id )
+                except MessageException:
+                    trans.response.status = 400
+                    return 'Wrong library dataset id: (%s)'  % str( dataset_id )
+                except ItemDeletionException:
+                    trans.response.status = 400
+                    return 'The item with library dataset id: (%s) is deleted'  % str( dataset_id )
+                except HTTPBadRequest, e:
+                    return 'http bad request' + str( e.err_msg )
+                except Exception, e:
+                    trans.response.status = 500
+                    return 'error of unknown kind' + str( e )
 
         if format in [ 'zip','tgz','tbz' ]:
                 error = False
https://bitbucket.org/galaxy/galaxy-central/commits/2b4877fc1024/
Changeset:   2b4877fc1024
User:        martenson
Date:        2013-12-17 19:35:08
Summary:     exception catching rewritten, it is now behaving better, not good but better than before
Affected #:  1 file
diff -r 16d53da082b89db1318a2a688aa8a18470b02f59 -r 2b4877fc1024180fadb5128187d447c0b84c7cd9 lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- a/lib/galaxy/webapps/galaxy/api/lda_datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -1,5 +1,5 @@
 """
-API operations on the dataset from library.
+API operations on the datasets from library.
 """
 import glob
 import logging
@@ -35,6 +35,7 @@
         try:
             dataset = self.get_library_dataset( trans, id = id, check_ownership=False, check_accessible=True )
         except Exception, e:
+            trans.response.status = 500
             return str( e )
         try:
             # Default: return dataset as dict.
@@ -43,24 +44,22 @@
             rval = "Error in dataset API at listing contents: " + str( e )
             log.error( rval + ": %s" % str(e), exc_info=True )
             trans.response.status = 500
+            return "Error in dataset API at listing contents: " + str( e )
 
         rval['id'] = trans.security.encode_id(rval['id']);
         rval['ldda_id'] = trans.security.encode_id(rval['ldda_id']);
         rval['folder_id'] = 'f' + trans.security.encode_id(rval['folder_id'])
-
+        trans.response.status = 200
         return rval
 
     @web.expose
     def download( self, trans, format, **kwd ):
         """
-        POST /api/libraries/datasets/download/{format}
-        POST data: ldda_ids = []
-        Downloads dataset(s) in the requested format.
+        GET /api/libraries/datasets/download/{format}
+        GET multiple params: ldda_ids = []
+        Downloads dataset(s) in the requested format or plain.
         """
         lddas = []
-#         is_admin = trans.user_is_admin()
-#         current_user_roles = trans.get_current_user_roles()
-
         datasets_to_download = kwd['ldda_ids%5B%5D']
 
         if ( datasets_to_download != None ):
@@ -111,15 +110,13 @@
                         archive = StreamBall( 'w|bz2' )
                         outext = 'tbz2'
                 except ( OSError, zipfile.BadZipfile ):
-                    error = True
                     log.exception( "Unable to create archive for download" )
-                    message = "Unable to create archive for download, please report this error"
-                    status = 'error'
+                    trans.response.status = 500
+                    return "Unable to create archive for download, please report this error"
                 except:
-                     error = True
                      log.exception( "Unexpected error %s in create archive for download" % sys.exc_info()[0] )
-                     message = "Unable to create archive for download, please report - %s" % sys.exc_info()[0]
-                     status = 'error'
+                     trans.response.status = 500
+                     return "Unable to create archive for download, please report - %s" % sys.exc_info()[0]
                 if not error:
                     composite_extensions = trans.app.datatypes_registry.get_composite_extensions()
                     seen = []
@@ -150,11 +147,9 @@
                             try:
                                 archive.add(ldda.dataset.file_name,zpath) # add the primary of a composite set
                             except IOError:
-                                error = True
                                 log.exception( "Unable to add composite parent %s to temporary library download archive" % ldda.dataset.file_name)
-                                message = "Unable to create archive for download, please report this error"
-                                status = 'error'
-                                continue
+                                trans.response.status = 500
+                                return "Unable to create archive for download, please report this error"
                             flist = glob.glob(os.path.join(ldda.dataset.extra_files_path,'*.*')) # glob returns full paths
                             for fpath in flist:
                                 efp,fname = os.path.split(fpath)
@@ -163,19 +158,16 @@
                                 try:
                                     archive.add( fpath,fname )
                                 except IOError:
-                                    error = True
                                     log.exception( "Unable to add %s to temporary library download archive %s" % (fname,outfname))
-                                    message = "Unable to create archive for download, please report this error"
-                                    status = 'error'
-                                    continue
+                                    trans.response.status = 500
+                                    return "Unable to create archive for download, please report this error"
                         else: # simple case
                             try:
                                 archive.add( ldda.dataset.file_name, path )
                             except IOError:
-                                error = True
                                 log.exception( "Unable to write %s to temporary library download archive" % ldda.dataset.file_name)
-                                message = "Unable to create archive for download, please report this error"
-                                status = 'error'
+                                trans.response.status = 500
+                                return "Unable to create archive for download, please report this error"
                     if not error:
                         lname = 'selected_dataset'
                         fname = lname.replace( ' ', '_' ) + '_files'
@@ -186,15 +178,18 @@
                             archive = util.streamball.ZipBall(tmpf, tmpd)
                             archive.wsgi_status = trans.response.wsgi_status()
                             archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            trans.response.status = 200
                             return archive.stream
                         else:
                             trans.response.set_content_type( "application/x-tar" )
                             trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
                             archive.wsgi_status = trans.response.wsgi_status()
                             archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                            trans.response.status = 200
                             return archive.stream
         elif format == 'uncompressed':
             if len(lddas) != 1:
+                trans.response.status = 400
                 return 'Wrong request'
             else:
                 single_dataset = lddas[0]
@@ -206,8 +201,11 @@
                 fname = ''.join( c in valid_chars and c or '_' for c in fname )[ 0:150 ]
                 trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s"' % fname
                 try:
+                    trans.response.status = 200
                     return open( single_dataset.file_name )
                 except:
+                    trans.response.status = 500
                     return 'This dataset contains no content'
         else:
-            return 'Wrong format';
+            trans.response.status = 400
+            return 'Wrong format parameter specified';
https://bitbucket.org/galaxy/galaxy-central/commits/6f89cbaeb29c/
Changeset:   6f89cbaeb29c
User:        martenson
Date:        2013-12-17 21:58:40
Summary:     fix for errors thrown when requested underlying file of dataset does not exist; even better exception handling for ldda-archive downloading
Affected #:  2 files
diff -r 2b4877fc1024180fadb5128187d447c0b84c7cd9 -r 6f89cbaeb29c33ae8d662d84965bf3bdee7f179d lib/galaxy/util/streamball.py
--- a/lib/galaxy/util/streamball.py
+++ b/lib/galaxy/util/streamball.py
@@ -3,6 +3,7 @@
 """
 import os
 import logging, tarfile
+from galaxy.exceptions import ObjectNotFound
 
 log = logging.getLogger( __name__ )
 
@@ -14,8 +15,12 @@
         self.mode = mode
         self.wsgi_status = None
         self.wsgi_headeritems = None
-    def add( self, file, relpath ):
-        self.members[file] = relpath
+    def add( self, file, relpath, check_file=False):
+        if check_file and len(file)>0:
+            if not os.path.isfile(file):
+                raise ObjectNotFound
+        else:
+            self.members[file] = relpath
     def stream( self, environ, start_response ):
         response_write = start_response( self.wsgi_status, self.wsgi_headeritems )
         class tarfileobj:
diff -r 2b4877fc1024180fadb5128187d447c0b84c7cd9 -r 6f89cbaeb29c33ae8d662d84965bf3bdee7f179d lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- a/lib/galaxy/webapps/galaxy/api/lda_datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -14,7 +14,7 @@
 import urllib2
 import zipfile
 from paste.httpexceptions import HTTPBadRequest
-from galaxy.exceptions import ItemAccessibilityException, MessageException, ItemDeletionException
+from galaxy.exceptions import ItemAccessibilityException, MessageException, ItemDeletionException, ObjectNotFound
 from galaxy.security import Action
 from galaxy import util, web
 from galaxy.util.streamball import StreamBall
@@ -84,7 +84,7 @@
                     return 'error of unknown kind' + str( e )
 
         if format in [ 'zip','tgz','tbz' ]:
-                error = False
+                # error = False
                 killme = string.punctuation + string.whitespace
                 trantab = string.maketrans(killme,'_'*len(killme))
                 try:
@@ -117,12 +117,12 @@
                      log.exception( "Unexpected error %s in create archive for download" % sys.exc_info()[0] )
                      trans.response.status = 500
                      return "Unable to create archive for download, please report - %s" % sys.exc_info()[0]
-                if not error:
+                if True:
                     composite_extensions = trans.app.datatypes_registry.get_composite_extensions()
                     seen = []
                     for ldda in lddas:
-                        if ldda.dataset.state in [ 'new', 'upload', 'queued', 'running', 'empty', 'discarded' ]:
-                            continue
+                        # if ldda.dataset.state in [ 'new', 'upload', 'queued', 'running', 'empty', 'discarded' ]:
+                            # continue
                         ext = ldda.extension
                         is_composite = ext in composite_extensions
                         path = ""
@@ -135,6 +135,7 @@
                             path = os.path.join( parent_folder.name, path )
                             parent_folder = parent_folder.parent
                         path += ldda.name
+
                         while path in seen:
                             path += '_'
                         seen.append( path )
@@ -145,30 +146,61 @@
                             if zpathext == '':
                                 zpath = '%s.html' % zpath # fake the real nature of the html file
                             try:
-                                archive.add(ldda.dataset.file_name,zpath) # add the primary of a composite set
+                                if format=='zip':
+                                    archive.add( ldda.dataset.file_name, zpath ) # add the primary of a composite set
+                                else:                                    
+                                    archive.add( ldda.dataset.file_name, zpath, check_file=True ) # add the primary of a composite set
                             except IOError:
                                 log.exception( "Unable to add composite parent %s to temporary library download archive" % ldda.dataset.file_name)
                                 trans.response.status = 500
                                 return "Unable to create archive for download, please report this error"
+                            except ObjectNotFound:
+                                log.exception( "Requested dataset %s does not exist on the host." % ldda.dataset.file_name )
+                                trans.response.status = 500
+                                return "Requested dataset does not exist on the host."
+                            except:
+                                trans.response.status = 500
+                                return "Unknown error, please report this error"                                
                             flist = glob.glob(os.path.join(ldda.dataset.extra_files_path,'*.*')) # glob returns full paths
                             for fpath in flist:
                                 efp,fname = os.path.split(fpath)
                                 if fname > '':
                                     fname = fname.translate(trantab)
                                 try:
-                                    archive.add( fpath,fname )
+                                    if format=='zip':
+                                        archive.add( fpath,fname )
+                                    else:
+                                        archive.add( fpath,fname, check_file=True  )
                                 except IOError:
                                     log.exception( "Unable to add %s to temporary library download archive %s" % (fname,outfname))
                                     trans.response.status = 500
                                     return "Unable to create archive for download, please report this error"
+                                except ObjectNotFound:
+                                    log.exception( "Requested dataset %s does not exist on the host." % fpath )
+                                    trans.response.status = 500
+                                    return "Requested dataset does not exist on the host."                                    
+                                except:
+                                    trans.response.status = 500
+                                    return "Unknown error, please report this error"
                         else: # simple case
                             try:
-                                archive.add( ldda.dataset.file_name, path )
+                                if format=='zip':
+                                    archive.add( ldda.dataset.file_name, path )
+                                else:
+                                    archive.add( ldda.dataset.file_name, path, check_file=True  )
                             except IOError:
                                 log.exception( "Unable to write %s to temporary library download archive" % ldda.dataset.file_name)
                                 trans.response.status = 500
                                 return "Unable to create archive for download, please report this error"
-                    if not error:
+                            except ObjectNotFound:
+                                log.exception( "Requested dataset %s does not exist on the host." % ldda.dataset.file_name )
+                                trans.response.status = 500
+                                return "Requested dataset does not exist on the host."
+                            except:
+                                trans.response.status = 500
+                                return "Unknown error, please report this error"
+
+                    if True:
                         lname = 'selected_dataset'
                         fname = lname.replace( ' ', '_' ) + '_files'
                         if format == 'zip':
https://bitbucket.org/galaxy/galaxy-central/commits/8090554de504/
Changeset:   8090554de504
User:        martenson
Date:        2013-12-18 17:41:24
Summary:     minor cosmetic changes
Affected #:  2 files
diff -r 6f89cbaeb29c33ae8d662d84965bf3bdee7f179d -r 8090554de504ffa05e744fdca051b9350aeae93f static/scripts/utils/galaxy.uploadbox.js
--- a/static/scripts/utils/galaxy.uploadbox.js
+++ b/static/scripts/utils/galaxy.uploadbox.js
@@ -353,3 +353,4 @@
         };
     }
 })(jQuery);
+
diff -r 6f89cbaeb29c33ae8d662d84965bf3bdee7f179d -r 8090554de504ffa05e744fdca051b9350aeae93f static/style/Gruntfile.js
--- a/static/style/Gruntfile.js
+++ b/static/style/Gruntfile.js
@@ -6,7 +6,7 @@
   var theme = grunt.option( 'theme', 'blue' );
 
   var out = 'blue'
-  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'trackster'];
+  var lessFiles = [ 'base', 'autocomplete_tagging', 'embed_item', 'iphone', 'masthead', 'library', 'trackster' ];
 
   var _ = grunt.util._; 
   var fmt = _.sprintf; 
https://bitbucket.org/galaxy/galaxy-central/commits/00fb5740d971/
Changeset:   00fb5740d971
User:        martenson
Date:        2013-12-19 18:44:34
Summary:     refactor + docs
Affected #:  2 files
diff -r 8090554de504ffa05e744fdca051b9350aeae93f -r 00fb5740d971318d4384e69fc2fcc6b2318b2d44 doc/source/lib/galaxy.webapps.galaxy.api.rst
--- a/doc/source/lib/galaxy.webapps.galaxy.api.rst
+++ b/doc/source/lib/galaxy.webapps.galaxy.api.rst
@@ -297,6 +297,14 @@
     :undoc-members:
     :show-inheritance:
 
+:mod:`lda_datasets` Module
+--------------------------
+
+.. automodule:: galaxy.webapps.galaxy.api.lda_datasets
+    :members:
+    :undoc-members:
+    :show-inheritance:
+
 :mod:`libraries` Module
 -----------------------
 
diff -r 8090554de504ffa05e744fdca051b9350aeae93f -r 00fb5740d971318d4384e69fc2fcc6b2318b2d44 lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- a/lib/galaxy/webapps/galaxy/api/lda_datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -28,8 +28,17 @@
     @web.expose_api
     def show( self, trans, id, **kwd ):
         """
-        GET /api/libraries/datasets/{encoded_dataset_id}
-        Displays information about the dataset identified by the lda ID.
+        show( self, trans, id, **kwd )
+        * GET /api/libraries/datasets/{encoded_dataset_id}:
+            Displays information about the dataset identified by the encoded id.
+
+
+        :type   id:      an encoded id string
+        :param  id:      the encoded id of the dataset to query
+
+        :rtype:     dictionary
+        :returns:   detailed dataset information from
+            :func:`galaxy.web.base.controller.UsesVisualizationMixin.get_library_dataset.to_dict()`
         """
         # Get dataset.
         try:
@@ -55,9 +64,25 @@
     @web.expose
     def download( self, trans, format, **kwd ):
         """
-        GET /api/libraries/datasets/download/{format}
-        GET multiple params: ldda_ids = []
-        Downloads dataset(s) in the requested format or plain.
+        download( self, trans, format, **kwd )
+        * GET /api/libraries/datasets/download/{format}
+
+        .. code-block::
+            example:
+            GET localhost:8080/api/libraries/datasets/download/tbz?ldda_ids%255B%255D=a0d84b45643a2678&ldda_ids%255B%255D=fe38c84dcd46c828
+
+        :type   format:      string
+        :param  format:      string representing requested archive format
+
+        .. note:: supported formats are: zip, tgz, tbz, uncompressed
+
+        :type   lddas[]:      an array
+        :param  lddas[]:      an array of encoded ids
+
+        :rtype:   file
+        :returns: either archive with the requested datasets packed inside or a single uncompressed dataset
+
+        :raises: MessageException, ItemDeletionException, ItemAccessibilityException, HTTPBadRequest, OSError, IOError, ObjectNotFound
         """
         lddas = []
         datasets_to_download = kwd['ldda_ids%5B%5D']
@@ -117,108 +142,101 @@
                      log.exception( "Unexpected error %s in create archive for download" % sys.exc_info()[0] )
                      trans.response.status = 500
                      return "Unable to create archive for download, please report - %s" % sys.exc_info()[0]
-                if True:
-                    composite_extensions = trans.app.datatypes_registry.get_composite_extensions()
-                    seen = []
-                    for ldda in lddas:
-                        # if ldda.dataset.state in [ 'new', 'upload', 'queued', 'running', 'empty', 'discarded' ]:
-                            # continue
-                        ext = ldda.extension
-                        is_composite = ext in composite_extensions
-                        path = ""
-                        parent_folder = ldda.library_dataset.folder
-                        while parent_folder is not None:
-                            # Exclude the now-hidden "root folder"
-                            if parent_folder.parent is None:
-                                path = os.path.join( parent_folder.library_root[0].name, path )
-                                break
-                            path = os.path.join( parent_folder.name, path )
-                            parent_folder = parent_folder.parent
-                        path += ldda.name
-
-                        while path in seen:
-                            path += '_'
-                        seen.append( path )
-                        zpath = os.path.split(path)[-1] # comes as base_name/fname
-                        outfname,zpathext = os.path.splitext(zpath)
-                        if is_composite:
-                            # need to add all the components from the extra_files_path to the zip
-                            if zpathext == '':
-                                zpath = '%s.html' % zpath # fake the real nature of the html file
+                composite_extensions = trans.app.datatypes_registry.get_composite_extensions()
+                seen = []
+                for ldda in lddas:
+                    ext = ldda.extension
+                    is_composite = ext in composite_extensions
+                    path = ""
+                    parent_folder = ldda.library_dataset.folder
+                    while parent_folder is not None:
+                        # Exclude the now-hidden "root folder"
+                        if parent_folder.parent is None:
+                            path = os.path.join( parent_folder.library_root[0].name, path )
+                            break
+                        path = os.path.join( parent_folder.name, path )
+                        parent_folder = parent_folder.parent
+                    path += ldda.name
+                    while path in seen:
+                        path += '_'
+                    seen.append( path )
+                    zpath = os.path.split(path)[-1] # comes as base_name/fname
+                    outfname,zpathext = os.path.splitext(zpath)
+                    if is_composite: # need to add all the components from the extra_files_path to the zip
+                        if zpathext == '':
+                            zpath = '%s.html' % zpath # fake the real nature of the html file
+                        try:
+                            if format=='zip':
+                                archive.add( ldda.dataset.file_name, zpath ) # add the primary of a composite set
+                            else:                                    
+                                archive.add( ldda.dataset.file_name, zpath, check_file=True ) # add the primary of a composite set
+                        except IOError:
+                            log.exception( "Unable to add composite parent %s to temporary library download archive" % ldda.dataset.file_name)
+                            trans.response.status = 500
+                            return "Unable to create archive for download, please report this error"
+                        except ObjectNotFound:
+                            log.exception( "Requested dataset %s does not exist on the host." % ldda.dataset.file_name )
+                            trans.response.status = 500
+                            return "Requested dataset does not exist on the host."
+                        except:
+                            trans.response.status = 500
+                            return "Unknown error, please report this error"                                
+                        flist = glob.glob(os.path.join(ldda.dataset.extra_files_path,'*.*')) # glob returns full paths
+                        for fpath in flist:
+                            efp,fname = os.path.split(fpath)
+                            if fname > '':
+                                fname = fname.translate(trantab)
                             try:
                                 if format=='zip':
-                                    archive.add( ldda.dataset.file_name, zpath ) # add the primary of a composite set
-                                else:                                    
-                                    archive.add( ldda.dataset.file_name, zpath, check_file=True ) # add the primary of a composite set
+                                    archive.add( fpath,fname )
+                                else:
+                                    archive.add( fpath,fname, check_file=True  )
                             except IOError:
-                                log.exception( "Unable to add composite parent %s to temporary library download archive" % ldda.dataset.file_name)
+                                log.exception( "Unable to add %s to temporary library download archive %s" % (fname,outfname))
                                 trans.response.status = 500
                                 return "Unable to create archive for download, please report this error"
                             except ObjectNotFound:
-                                log.exception( "Requested dataset %s does not exist on the host." % ldda.dataset.file_name )
+                                log.exception( "Requested dataset %s does not exist on the host." % fpath )
                                 trans.response.status = 500
-                                return "Requested dataset does not exist on the host."
-                            except:
-                                trans.response.status = 500
-                                return "Unknown error, please report this error"                                
-                            flist = glob.glob(os.path.join(ldda.dataset.extra_files_path,'*.*')) # glob returns full paths
-                            for fpath in flist:
-                                efp,fname = os.path.split(fpath)
-                                if fname > '':
-                                    fname = fname.translate(trantab)
-                                try:
-                                    if format=='zip':
-                                        archive.add( fpath,fname )
-                                    else:
-                                        archive.add( fpath,fname, check_file=True  )
-                                except IOError:
-                                    log.exception( "Unable to add %s to temporary library download archive %s" % (fname,outfname))
-                                    trans.response.status = 500
-                                    return "Unable to create archive for download, please report this error"
-                                except ObjectNotFound:
-                                    log.exception( "Requested dataset %s does not exist on the host." % fpath )
-                                    trans.response.status = 500
-                                    return "Requested dataset does not exist on the host."                                    
-                                except:
-                                    trans.response.status = 500
-                                    return "Unknown error, please report this error"
-                        else: # simple case
-                            try:
-                                if format=='zip':
-                                    archive.add( ldda.dataset.file_name, path )
-                                else:
-                                    archive.add( ldda.dataset.file_name, path, check_file=True  )
-                            except IOError:
-                                log.exception( "Unable to write %s to temporary library download archive" % ldda.dataset.file_name)
-                                trans.response.status = 500
-                                return "Unable to create archive for download, please report this error"
-                            except ObjectNotFound:
-                                log.exception( "Requested dataset %s does not exist on the host." % ldda.dataset.file_name )
-                                trans.response.status = 500
-                                return "Requested dataset does not exist on the host."
+                                return "Requested dataset does not exist on the host."                                    
                             except:
                                 trans.response.status = 500
                                 return "Unknown error, please report this error"
-
-                    if True:
-                        lname = 'selected_dataset'
-                        fname = lname.replace( ' ', '_' ) + '_files'
-                        if format == 'zip':
-                            archive.close()
-                            trans.response.set_content_type( "application/octet-stream" )
-                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
-                            archive = util.streamball.ZipBall(tmpf, tmpd)
-                            archive.wsgi_status = trans.response.wsgi_status()
-                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
-                            trans.response.status = 200
-                            return archive.stream
-                        else:
-                            trans.response.set_content_type( "application/x-tar" )
-                            trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
-                            archive.wsgi_status = trans.response.wsgi_status()
-                            archive.wsgi_headeritems = trans.response.wsgi_headeritems()
-                            trans.response.status = 200
-                            return archive.stream
+                    else: # simple case
+                        try:
+                            if format=='zip':
+                                archive.add( ldda.dataset.file_name, path )
+                            else:
+                                archive.add( ldda.dataset.file_name, path, check_file=True  )
+                        except IOError:
+                            log.exception( "Unable to write %s to temporary library download archive" % ldda.dataset.file_name)
+                            trans.response.status = 500
+                            return "Unable to create archive for download, please report this error"
+                        except ObjectNotFound:
+                            log.exception( "Requested dataset %s does not exist on the host." % ldda.dataset.file_name )
+                            trans.response.status = 500
+                            return "Requested dataset does not exist on the host."
+                        except:
+                            trans.response.status = 500
+                            return "Unknown error, please report this error"
+                lname = 'selected_dataset'
+                fname = lname.replace( ' ', '_' ) + '_files'
+                if format == 'zip':
+                    archive.close()
+                    trans.response.set_content_type( "application/octet-stream" )
+                    trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                    archive = util.streamball.ZipBall(tmpf, tmpd)
+                    archive.wsgi_status = trans.response.wsgi_status()
+                    archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                    trans.response.status = 200
+                    return archive.stream
+                else:
+                    trans.response.set_content_type( "application/x-tar" )
+                    trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                    archive.wsgi_status = trans.response.wsgi_status()
+                    archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                    trans.response.status = 200
+                    return archive.stream
         elif format == 'uncompressed':
             if len(lddas) != 1:
                 trans.response.status = 400
https://bitbucket.org/galaxy/galaxy-central/commits/c1963ee06cb0/
Changeset:   c1963ee06cb0
User:        martenson
Date:        2013-12-19 18:48:22
Summary:     Merged in martenson/galaxy-central-marten (pull request #273)
Initial Data Libraries
Affected #:  24 files
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b doc/source/lib/galaxy.webapps.galaxy.api.rst
--- a/doc/source/lib/galaxy.webapps.galaxy.api.rst
+++ b/doc/source/lib/galaxy.webapps.galaxy.api.rst
@@ -302,6 +302,14 @@
     :undoc-members:
     :show-inheritance:
 
+:mod:`lda_datasets` Module
+--------------------------
+
+.. automodule:: galaxy.webapps.galaxy.api.lda_datasets
+    :members:
+    :undoc-members:
+    :show-inheritance:
+
 :mod:`libraries` Module
 -----------------------
 
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/model/__init__.py
--- a/lib/galaxy/model/__init__.py
+++ b/lib/galaxy/model/__init__.py
@@ -1827,7 +1827,7 @@
 class Library( object, Dictifiable ):
     permitted_actions = get_permitted_actions( filter='LIBRARY' )
     dict_collection_visible_keys = ( 'id', 'name' )
-    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis' )
+    dict_element_visible_keys = ( 'id', 'deleted', 'name', 'description', 'synopsis', 'root_folder_id' )
     def __init__( self, name=None, description=None, synopsis=None, root_folder=None ):
         self.name = name or "Unnamed library"
         self.description = description
@@ -1894,7 +1894,7 @@
         return name
 
 class LibraryFolder( object, Dictifiable ):
-    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build' )
+    dict_element_visible_keys = ( 'id', 'parent_id', 'name', 'description', 'item_count', 'genome_build', 'update_time' )
     def __init__( self, name=None, description=None, item_count=0, order_id=None ):
         self.name = name or "Unnamed folder"
         self.description = description
@@ -2060,6 +2060,7 @@
                      genome_build = ldda.dbkey,
                      misc_info = ldda.info,
                      misc_blurb = ldda.blurb,
+                     peek = ( lambda ldda: ldda.display_peek() if ldda.peek and ldda.peek != 'no peek' else None )( ldda ),
                      template_data = template_data )
         if ldda.dataset.uuid is None:
             rval['uuid'] = None
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/model/mapping.py
--- a/lib/galaxy/model/mapping.py
+++ b/lib/galaxy/model/mapping.py
@@ -1852,8 +1852,9 @@
     table = self.table
     trans = conn.begin()
     try:
-        next_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
-        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid + 1 ) )
+        current_hid = select( [table.c.hid_counter], table.c.id == self.id, for_update=True ).scalar()
+        next_hid = current_hid + 1
+        table.update( table.c.id == self.id ).execute( hid_counter = ( next_hid ) )
         trans.commit()
         return next_hid
     except:
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/util/streamball.py
--- a/lib/galaxy/util/streamball.py
+++ b/lib/galaxy/util/streamball.py
@@ -3,6 +3,7 @@
 """
 import os
 import logging, tarfile
+from galaxy.exceptions import ObjectNotFound
 
 log = logging.getLogger( __name__ )
 
@@ -14,8 +15,12 @@
         self.mode = mode
         self.wsgi_status = None
         self.wsgi_headeritems = None
-    def add( self, file, relpath ):
-        self.members[file] = relpath
+    def add( self, file, relpath, check_file=False):
+        if check_file and len(file)>0:
+            if not os.path.isfile(file):
+                raise ObjectNotFound
+        else:
+            self.members[file] = relpath
     def stream( self, environ, start_response ):
         response_write = start_response( self.wsgi_status, self.wsgi_headeritems )
         class tarfileobj:
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/webapps/galaxy/api/datasets.py
--- a/lib/galaxy/webapps/galaxy/api/datasets.py
+++ b/lib/galaxy/webapps/galaxy/api/datasets.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a dataset.
+API operations on the contents of a history dataset.
 """
 from galaxy import web
 from galaxy.visualization.data_providers.genome import FeatureLocationIndexDataProvider
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/webapps/galaxy/api/folder_contents.py
--- a/lib/galaxy/webapps/galaxy/api/folder_contents.py
+++ b/lib/galaxy/webapps/galaxy/api/folder_contents.py
@@ -1,5 +1,5 @@
 """
-API operations on the contents of a library.
+API operations on the contents of a folder.
 """
 import logging, os, string, shutil, urllib, re, socket
 from cgi import escape, FieldStorage
@@ -11,67 +11,122 @@
 log = logging.getLogger( __name__ )
 
 class FolderContentsController( BaseAPIController, UsesLibraryMixin, UsesLibraryMixinItems ):
+    """
+    Class controls retrieval, creation and updating of folder contents.
+    """
+
+    def load_folder_contents( self, trans, folder ):
+        """
+        Loads all contents of the folder (folders and data sets) but only in the first level.
+        """
+        current_user_roles = trans.get_current_user_roles()
+        is_admin = trans.user_is_admin()
+        content_items = []
+        for subfolder in folder.active_folders:
+            if not is_admin:
+                can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
+            if (is_admin or can_access) and not subfolder.deleted:
+                subfolder.api_type = 'folder'
+                content_items.append( subfolder )
+        for dataset in folder.datasets:
+            if not is_admin:
+                can_access = trans.app.security_agent.can_access_dataset( current_user_roles, dataset.library_dataset_dataset_association.dataset )
+            if (is_admin or can_access) and not dataset.deleted:
+                dataset.api_type = 'file'
+                content_items.append( dataset )
+        return content_items
 
     @web.expose_api
     def index( self, trans, folder_id, **kwd ):
         """
         GET /api/folders/{encoded_folder_id}/contents
         Displays a collection (list) of a folder's contents (files and folders).
-        The /api/library_contents/{encoded_library_id}/contents
-        lists everything in a library recursively, which is not what
-        we want here. We could add a parameter to use the recursive
-        style, but this is meant to act similar to an "ls" directory listing.
+        Encoded folder ID is prepended with 'F' if it is a folder as opposed to a data set which does not have it.
+        Full path is provided as a separate object in response providing data for breadcrumb path building.
         """
-        rval = []
+        folder_container = []
         current_user_roles = trans.get_current_user_roles()
 
-        def traverse( folder ):
-            admin = trans.user_is_admin()
-            rval = []
-            for subfolder in folder.active_folders:
-                if not admin:
-                    can_access, folder_ids = trans.app.security_agent.check_folder_contents( trans.user, current_user_roles, subfolder )
-                if (admin or can_access) and not subfolder.deleted:
-                    subfolder.api_type = 'folder'
-                    rval.append( subfolder )
-            for ld in folder.datasets:
-                if not admin:
-                    can_access = trans.app.security_agent.can_access_dataset( current_user_roles, ld.library_dataset_dataset_association.dataset )
-                if (admin or can_access) and not ld.deleted:
-                    ld.api_type = 'file'
-                    rval.append( ld )
-            return rval
-
-        try:
-            decoded_folder_id = trans.security.decode_id( folder_id[-16:] )
-        except TypeError:
-            trans.response.status = 400
-            return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
+        if ( folder_id.startswith( 'F' ) ):
+            try:
+                decoded_folder_id = trans.security.decode_id( folder_id[1:] )
+            except TypeError:
+                trans.response.status = 400
+                return "Malformed folder id ( %s ) specified, unable to decode." % str( folder_id )
 
         try:
             folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( decoded_folder_id )
-            parent_library = folder.parent_library
         except:
             folder = None
-            log.error( "FolderContentsController.index: Unable to retrieve folder %s"
-                      % folder_id )
+            log.error( "FolderContentsController.index: Unable to retrieve folder with ID: %s" % folder_id )
 
-        # TODO: Find the API's path to this folder if necessary.
-        # This was needed in recursive descent, but it's not needed
-        # for "ls"-style content checking:
-        if not folder or not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+        # We didn't find the folder or user does not have an access to it.
+        if not folder:
             trans.response.status = 400
             return "Invalid folder id ( %s ) specified." % str( folder_id )
+        
+        if not ( trans.user_is_admin() or trans.app.security_agent.can_access_library_item( current_user_roles, folder, trans.user ) ):
+            log.warning( "SECURITY: User (id: %s) without proper access rights is trying to load folder with ID of %s" % ( trans.user.id, folder.id ) )
+            trans.response.status = 400
+            return "Invalid folder id ( %s ) specified." % str( folder_id )
+        
+        path_to_root = []
+        def build_path ( folder ):
+            """
+            Search the path upwards recursively and load the whole route of names and ids for breadcrumb purposes.
+            """
+            path_to_root = []
+            # We are almost in root
+            if folder.parent_id is None:
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
+            else:
+            # We add the current folder and traverse up one folder.
+                path_to_root.append( ( 'F' + trans.security.encode_id( folder.id ), folder.name ) )
+                upper_folder = trans.sa_session.query( trans.app.model.LibraryFolder ).get( folder.parent_id )
+                path_to_root.extend( build_path( upper_folder ) )
+            return path_to_root
+            
+        # Return the reversed path so it starts with the library node.
+        full_path = build_path( folder )[::-1]
+        folder_container.append( dict( full_path = full_path ) )
+        
+        folder_contents = []
+        time_updated = ''
+        time_created = ''
+        # Go through every item in the folder and include its meta-data.
+        for content_item in self.load_folder_contents( trans, folder ):
+#             rval = content_item.to_dict()
+            return_item = {}
+            encoded_id = trans.security.encode_id( content_item.id )
+            time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+            time_created = content_item.create_time.strftime( "%Y-%m-%d %I:%M %p" )
+            
+            # For folder return also hierarchy values
+            if content_item.api_type == 'folder':
+                encoded_id = 'F' + encoded_id
+#                 time_updated = content_item.update_time.strftime( "%Y-%m-%d %I:%M %p" )
+                return_item.update ( dict ( item_count = content_item.item_count ) )
 
-        for content in traverse( folder ):
-            encoded_id = trans.security.encode_id( content.id )
-            if content.api_type == 'folder':
-                encoded_id = 'F' + encoded_id
-            rval.append( dict( id = encoded_id,
-                               type = content.api_type,
-                               name = content.name,
-                               url = url_for( 'folder_contents', folder_id=encoded_id ) ) )
-        return rval
+            if content_item.api_type == 'file':
+                library_dataset_dict = content_item.to_dict()
+                library_dataset_dict['data_type']
+                library_dataset_dict['file_size']
+                library_dataset_dict['date_uploaded']
+                return_item.update ( dict ( data_type = library_dataset_dict['data_type'],
+                                            file_size = library_dataset_dict['file_size'],
+                                            date_uploaded = library_dataset_dict['date_uploaded'] ) )
+
+            # For every item return also the default meta-data
+            return_item.update( dict( id = encoded_id,
+                               type = content_item.api_type,
+                               name = content_item.name,
+                               time_updated = time_updated,
+                               time_created = time_created
+                                ) )
+            folder_contents.append( return_item )
+        # Put the data in the container
+        folder_container.append( dict( folder_contents = folder_contents ) )
+        return folder_container
 
     @web.expose_api
     def show( self, trans, id, library_id, **kwd ):
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/webapps/galaxy/api/lda_datasets.py
--- /dev/null
+++ b/lib/galaxy/webapps/galaxy/api/lda_datasets.py
@@ -0,0 +1,261 @@
+"""
+API operations on the datasets from library.
+"""
+import glob
+import logging
+import operator
+import os
+import os.path
+import string
+import sys
+import tarfile
+import tempfile
+import urllib
+import urllib2
+import zipfile
+from paste.httpexceptions import HTTPBadRequest
+from galaxy.exceptions import ItemAccessibilityException, MessageException, ItemDeletionException, ObjectNotFound
+from galaxy.security import Action
+from galaxy import util, web
+from galaxy.util.streamball import StreamBall
+from galaxy.web.base.controller import BaseAPIController, UsesVisualizationMixin
+
+import logging
+log = logging.getLogger( __name__ )
+
+class LibraryDatasetsController( BaseAPIController, UsesVisualizationMixin ):
+
+    @web.expose_api
+    def show( self, trans, id, **kwd ):
+        """
+        show( self, trans, id, **kwd )
+        * GET /api/libraries/datasets/{encoded_dataset_id}:
+            Displays information about the dataset identified by the encoded id.
+
+
+        :type   id:      an encoded id string
+        :param  id:      the encoded id of the dataset to query
+
+        :rtype:     dictionary
+        :returns:   detailed dataset information from
+            :func:`galaxy.web.base.controller.UsesVisualizationMixin.get_library_dataset.to_dict()`
+        """
+        # Get dataset.
+        try:
+            dataset = self.get_library_dataset( trans, id = id, check_ownership=False, check_accessible=True )
+        except Exception, e:
+            trans.response.status = 500
+            return str( e )
+        try:
+            # Default: return dataset as dict.
+            rval = dataset.to_dict()
+        except Exception, e:
+            rval = "Error in dataset API at listing contents: " + str( e )
+            log.error( rval + ": %s" % str(e), exc_info=True )
+            trans.response.status = 500
+            return "Error in dataset API at listing contents: " + str( e )
+
+        rval['id'] = trans.security.encode_id(rval['id']);
+        rval['ldda_id'] = trans.security.encode_id(rval['ldda_id']);
+        rval['folder_id'] = 'f' + trans.security.encode_id(rval['folder_id'])
+        trans.response.status = 200
+        return rval
+
+    @web.expose
+    def download( self, trans, format, **kwd ):
+        """
+        download( self, trans, format, **kwd )
+        * GET /api/libraries/datasets/download/{format}
+
+        .. code-block::
+            example:
+            GET localhost:8080/api/libraries/datasets/download/tbz?ldda_ids%255B%255D=a0d84b45643a2678&ldda_ids%255B%255D=fe38c84dcd46c828
+
+        :type   format:      string
+        :param  format:      string representing requested archive format
+
+        .. note:: supported formats are: zip, tgz, tbz, uncompressed
+
+        :type   lddas[]:      an array
+        :param  lddas[]:      an array of encoded ids
+
+        :rtype:   file
+        :returns: either archive with the requested datasets packed inside or a single uncompressed dataset
+
+        :raises: MessageException, ItemDeletionException, ItemAccessibilityException, HTTPBadRequest, OSError, IOError, ObjectNotFound
+        """
+        lddas = []
+        datasets_to_download = kwd['ldda_ids%5B%5D']
+
+        if ( datasets_to_download != None ):
+            datasets_to_download = util.listify( datasets_to_download )
+            for dataset_id in datasets_to_download:
+                try:
+                    ldda = self.get_hda_or_ldda( trans, hda_ldda='ldda', dataset_id=dataset_id )
+                    lddas.append( ldda )
+                except ItemAccessibilityException:
+                    trans.response.status = 403
+                    return 'Insufficient rights to access library dataset with id: (%s)'  % str( dataset_id )
+                except MessageException:
+                    trans.response.status = 400
+                    return 'Wrong library dataset id: (%s)'  % str( dataset_id )
+                except ItemDeletionException:
+                    trans.response.status = 400
+                    return 'The item with library dataset id: (%s) is deleted'  % str( dataset_id )
+                except HTTPBadRequest, e:
+                    return 'http bad request' + str( e.err_msg )
+                except Exception, e:
+                    trans.response.status = 500
+                    return 'error of unknown kind' + str( e )
+
+        if format in [ 'zip','tgz','tbz' ]:
+                # error = False
+                killme = string.punctuation + string.whitespace
+                trantab = string.maketrans(killme,'_'*len(killme))
+                try:
+                    outext = 'zip'
+                    if format == 'zip':
+                        # Can't use mkstemp - the file must not exist first
+                        tmpd = tempfile.mkdtemp()
+                        util.umask_fix_perms( tmpd, trans.app.config.umask, 0777, self.app.config.gid )
+                        tmpf = os.path.join( tmpd, 'library_download.' + format )
+                        if trans.app.config.upstream_gzip:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_STORED, True )
+                        else:
+                            archive = zipfile.ZipFile( tmpf, 'w', zipfile.ZIP_DEFLATED, True )
+                        archive.add = lambda x, y: archive.write( x, y.encode('CP437') )
+                    elif format == 'tgz':
+                        if trans.app.config.upstream_gzip:
+                            archive = StreamBall( 'w|' )
+                            outext = 'tar'
+                        else:
+                            archive = StreamBall( 'w|gz' )
+                            outext = 'tgz'
+                    elif format == 'tbz':
+                        archive = StreamBall( 'w|bz2' )
+                        outext = 'tbz2'
+                except ( OSError, zipfile.BadZipfile ):
+                    log.exception( "Unable to create archive for download" )
+                    trans.response.status = 500
+                    return "Unable to create archive for download, please report this error"
+                except:
+                     log.exception( "Unexpected error %s in create archive for download" % sys.exc_info()[0] )
+                     trans.response.status = 500
+                     return "Unable to create archive for download, please report - %s" % sys.exc_info()[0]
+                composite_extensions = trans.app.datatypes_registry.get_composite_extensions()
+                seen = []
+                for ldda in lddas:
+                    ext = ldda.extension
+                    is_composite = ext in composite_extensions
+                    path = ""
+                    parent_folder = ldda.library_dataset.folder
+                    while parent_folder is not None:
+                        # Exclude the now-hidden "root folder"
+                        if parent_folder.parent is None:
+                            path = os.path.join( parent_folder.library_root[0].name, path )
+                            break
+                        path = os.path.join( parent_folder.name, path )
+                        parent_folder = parent_folder.parent
+                    path += ldda.name
+                    while path in seen:
+                        path += '_'
+                    seen.append( path )
+                    zpath = os.path.split(path)[-1] # comes as base_name/fname
+                    outfname,zpathext = os.path.splitext(zpath)
+                    if is_composite: # need to add all the components from the extra_files_path to the zip
+                        if zpathext == '':
+                            zpath = '%s.html' % zpath # fake the real nature of the html file
+                        try:
+                            if format=='zip':
+                                archive.add( ldda.dataset.file_name, zpath ) # add the primary of a composite set
+                            else:                                    
+                                archive.add( ldda.dataset.file_name, zpath, check_file=True ) # add the primary of a composite set
+                        except IOError:
+                            log.exception( "Unable to add composite parent %s to temporary library download archive" % ldda.dataset.file_name)
+                            trans.response.status = 500
+                            return "Unable to create archive for download, please report this error"
+                        except ObjectNotFound:
+                            log.exception( "Requested dataset %s does not exist on the host." % ldda.dataset.file_name )
+                            trans.response.status = 500
+                            return "Requested dataset does not exist on the host."
+                        except:
+                            trans.response.status = 500
+                            return "Unknown error, please report this error"                                
+                        flist = glob.glob(os.path.join(ldda.dataset.extra_files_path,'*.*')) # glob returns full paths
+                        for fpath in flist:
+                            efp,fname = os.path.split(fpath)
+                            if fname > '':
+                                fname = fname.translate(trantab)
+                            try:
+                                if format=='zip':
+                                    archive.add( fpath,fname )
+                                else:
+                                    archive.add( fpath,fname, check_file=True  )
+                            except IOError:
+                                log.exception( "Unable to add %s to temporary library download archive %s" % (fname,outfname))
+                                trans.response.status = 500
+                                return "Unable to create archive for download, please report this error"
+                            except ObjectNotFound:
+                                log.exception( "Requested dataset %s does not exist on the host." % fpath )
+                                trans.response.status = 500
+                                return "Requested dataset does not exist on the host."                                    
+                            except:
+                                trans.response.status = 500
+                                return "Unknown error, please report this error"
+                    else: # simple case
+                        try:
+                            if format=='zip':
+                                archive.add( ldda.dataset.file_name, path )
+                            else:
+                                archive.add( ldda.dataset.file_name, path, check_file=True  )
+                        except IOError:
+                            log.exception( "Unable to write %s to temporary library download archive" % ldda.dataset.file_name)
+                            trans.response.status = 500
+                            return "Unable to create archive for download, please report this error"
+                        except ObjectNotFound:
+                            log.exception( "Requested dataset %s does not exist on the host." % ldda.dataset.file_name )
+                            trans.response.status = 500
+                            return "Requested dataset does not exist on the host."
+                        except:
+                            trans.response.status = 500
+                            return "Unknown error, please report this error"
+                lname = 'selected_dataset'
+                fname = lname.replace( ' ', '_' ) + '_files'
+                if format == 'zip':
+                    archive.close()
+                    trans.response.set_content_type( "application/octet-stream" )
+                    trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                    archive = util.streamball.ZipBall(tmpf, tmpd)
+                    archive.wsgi_status = trans.response.wsgi_status()
+                    archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                    trans.response.status = 200
+                    return archive.stream
+                else:
+                    trans.response.set_content_type( "application/x-tar" )
+                    trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s.%s"' % (fname,outext)
+                    archive.wsgi_status = trans.response.wsgi_status()
+                    archive.wsgi_headeritems = trans.response.wsgi_headeritems()
+                    trans.response.status = 200
+                    return archive.stream
+        elif format == 'uncompressed':
+            if len(lddas) != 1:
+                trans.response.status = 400
+                return 'Wrong request'
+            else:
+                single_dataset = lddas[0]
+                trans.response.set_content_type( single_dataset.get_mime() )
+                fStat = os.stat( ldda.file_name )
+                trans.response.headers[ 'Content-Length' ] = int( fStat.st_size )
+                valid_chars = '.,^_-()[]0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
+                fname = ldda.name
+                fname = ''.join( c in valid_chars and c or '_' for c in fname )[ 0:150 ]
+                trans.response.headers[ "Content-Disposition" ] = 'attachment; filename="%s"' % fname
+                try:
+                    trans.response.status = 200
+                    return open( single_dataset.file_name )
+                except:
+                    trans.response.status = 500
+                    return 'This dataset contains no content'
+        else:
+            trans.response.status = 400
+            return 'Wrong format parameter specified';
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/webapps/galaxy/api/libraries.py
--- a/lib/galaxy/webapps/galaxy/api/libraries.py
+++ b/lib/galaxy/webapps/galaxy/api/libraries.py
@@ -49,9 +49,10 @@
                            trans.model.Library.table.c.id.in_( accessible_restricted_library_ids ) ) )
         rval = []
         for library in query:
-            item = library.to_dict()
+            item = library.to_dict( view='element' )
             item['url'] = url_for( route, id=trans.security.encode_id( library.id ) )
-            item['id'] = trans.security.encode_id( item['id'] )
+            item['id'] = 'F' + trans.security.encode_id( item['id'] )
+            item['root_folder_id'] = 'F' + trans.security.encode_id( item['root_folder_id'] )
             rval.append( item )
         return rval
 
@@ -131,6 +132,9 @@
         rval['name'] = name
         rval['id'] = encoded_id
         return rval
+    
+    def edit( self, trans, payload, **kwd  ):
+        return "Not implemented yet"
 
     @web.expose_api
     def delete( self, trans, id, **kwd ):
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/webapps/galaxy/buildapp.py
--- a/lib/galaxy/webapps/galaxy/buildapp.py
+++ b/lib/galaxy/webapps/galaxy/buildapp.py
@@ -46,12 +46,6 @@
     atexit.register( app.shutdown )
     # Create the universe WSGI application
     webapp = GalaxyWebApplication( app, session_cookie='galaxysession', name='galaxy' )
-    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
-    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
-                      controller='admin_toolshed',
-                      action='display_image_in_repository',
-                      repository_id=None,
-                      image_file=None )
     webapp.add_ui_controllers( 'galaxy.webapps.galaxy.controllers', app )
     # Force /history to go to /root/history -- needed since the tests assume this
     webapp.add_route( '/history', controller='root', action='history' )
@@ -75,22 +69,12 @@
     webapp.add_route( '/u/:username/v/:slug', controller='visualization', action='display_by_username_and_slug' )
     webapp.add_route( '/search', controller='search', action='index' )
 
-    # Add the web API
+    # ================
+    # =====  API =====
+    # ================
+
     webapp.add_api_controllers( 'galaxy.webapps.galaxy.api', app )
-    # The /folders section is experimental at this point:
-    log.debug( "app.config.api_folders: %s" % app.config.api_folders )
-    webapp.mapper.resource( 'folder', 'folders', path_prefix='/api' )
-    webapp.mapper.resource( 'content', 'contents',
-                                controller='folder_contents',
-                                name_prefix='folder_',
-                                path_prefix='/api/folders/:folder_id',
-                                parent_resources=dict( member_name='folder', collection_name='folders' ) )
-    webapp.mapper.resource( 'content',
-                                'contents',
-                                controller='library_contents',
-                                name_prefix='library_',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
+
     webapp.mapper.resource( 'content',
                                 'contents',
                                 controller='history_contents',
@@ -102,10 +86,6 @@
                               controller="datasets",
                               action="display",
                               conditions=dict(method=["GET"]))
-    webapp.mapper.resource( 'permission',
-                                'permissions',
-                                path_prefix='/api/libraries/:library_id',
-                                parent_resources=dict( member_name='library', collection_name='libraries' ) )
     webapp.mapper.resource( 'user',
                                 'users',
                                 controller='group_users',
@@ -127,11 +107,6 @@
     _add_item_tags_controller( webapp,
                                name_prefix="workflow_",
                                path_prefix='/api/workflows/:workflow_id' )
-
-    _add_item_extended_metadata_controller( webapp,
-                               name_prefix="library_dataset_",
-                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
-
     _add_item_annotation_controller( webapp,
                                name_prefix="history_content_",
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
@@ -141,7 +116,6 @@
     _add_item_annotation_controller( webapp,
                                name_prefix="workflow_",
                                path_prefix='/api/workflows/:workflow_id' )
-
     _add_item_provenance_controller( webapp,
                                name_prefix="history_content_",
                                path_prefix='/api/histories/:history_id/contents/:history_content_id' )
@@ -170,6 +144,7 @@
                             parent_resources=dict( member_name='datatype', collection_name='datatypes' ) )
     #webapp.mapper.connect( 'run_workflow', '/api/workflow/{workflow_id}/library/{library_id}', controller='workflows', action='run', workflow_id=None, library_id=None, conditions=dict(method=["GET"]) )
     webapp.mapper.resource( 'search', 'search', path_prefix='/api' )
+    
 
     # add as a non-ATOM API call to support the notion of a 'current/working' history unique to the history resource
     webapp.mapper.connect( "set_as_current", "/api/histories/{id}/set_as_current",
@@ -188,6 +163,64 @@
     webapp.mapper.connect("workflow_dict", '/api/workflows/{workflow_id}/download', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
     # Preserve the following download route for now for dependent applications  -- deprecate at some point
     webapp.mapper.connect("workflow_dict", '/api/workflows/download/{workflow_id}', controller='workflows', action='workflow_dict', conditions=dict(method=['GET']))
+
+    # =======================
+    # ===== LIBRARY API =====
+    # =======================
+    
+    webapp.mapper.connect( 'show_lda_item',
+                           '/api/libraries/datasets/:id', 
+                           controller='lda_datasets', 
+                           action='show', 
+                           conditions=dict( method=[ "GET" ] ) )
+
+    webapp.mapper.connect( 'download_lda_items', 
+                           '/api/libraries/datasets/download/:format', 
+                           controller='lda_datasets', 
+                           action='download', 
+                           conditions=dict( method=[ "POST", "GET" ] ) )
+
+    webapp.mapper.resource_with_deleted( 'library',
+                                         'libraries', 
+                                         path_prefix='/api' )
+    webapp.mapper.resource( 'folder', 
+                            'folders', 
+                            path_prefix='/api' )
+
+    webapp.mapper.resource( 'content', 
+                            'contents',
+                            controller='folder_contents',
+                            name_prefix='folder_',
+                            path_prefix='/api/folders/:folder_id',
+                            parent_resources=dict( member_name='folder', collection_name='folders' ) )
+
+    webapp.mapper.resource( 'content',
+                            'contents',
+                            controller='library_contents',
+                            name_prefix='library_',
+                            path_prefix='/api/libraries/:library_id',
+                            parent_resources=dict( member_name='library', collection_name='libraries' ) )
+
+    webapp.mapper.resource( 'permission',
+                            'permissions',
+                            path_prefix='/api/libraries/:library_id',
+                            parent_resources=dict( member_name='library', collection_name='libraries' ) )
+    
+    _add_item_extended_metadata_controller( webapp,
+                               name_prefix="library_dataset_",
+                               path_prefix='/api/libraries/:library_id/contents/:library_content_id' )
+    
+    # ====================
+    # ===== TOOLSHED =====
+    # ====================
+    
+    # Handle displaying tool help images and README file images contained in repositories installed from the tool shed.
+    webapp.add_route( '/admin_toolshed/static/images/:repository_id/:image_file',
+                      controller='admin_toolshed',
+                      action='display_image_in_repository',
+                      repository_id=None,
+                      image_file=None )
+    
     # Galaxy API for tool shed features.
     webapp.mapper.resource( 'tool_shed_repository',
                             'tool_shed_repositories',
@@ -201,6 +234,7 @@
                             path_prefix='/api',
                             new={ 'install_repository_revision' : 'POST' },
                             parent_resources=dict( member_name='tool_shed_repository', collection_name='tool_shed_repositories' ) )
+
     # Connect logger from app
     if app.trace_logger:
         webapp.trace_logger = app.trace_logger
@@ -221,7 +255,7 @@
         galaxy.model.mapping.metadata.engine.connection_provider._pool.dispose()
     except:
         pass
-    # Close any pooled database connections before forking
+     # Close any pooled database connections before forking
     try:
         galaxy.model.tool_shed_install.mapping.metadata.engine.connection_provider._pool.dispose()
     except:
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/webapps/galaxy/controllers/library.py
--- a/lib/galaxy/webapps/galaxy/controllers/library.py
+++ b/lib/galaxy/webapps/galaxy/controllers/library.py
@@ -76,6 +76,17 @@
 
     library_list_grid = LibraryListGrid()
 
+    
+    @web.expose
+    def list( self, trans, **kwd ):
+        params = util.Params( kwd )
+        # define app configuration for generic mako template
+        app = {
+            'jscript'       : "galaxy.library"
+        }
+        # fill template
+        return trans.fill_template('galaxy.panels.mako', config = {'app' : app})
+
     @web.expose
     def index( self, trans, **kwd ):
         params = util.Params( kwd )
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b lib/galaxy/webapps/galaxy/controllers/user.py
--- a/lib/galaxy/webapps/galaxy/controllers/user.py
+++ b/lib/galaxy/webapps/galaxy/controllers/user.py
@@ -826,19 +826,22 @@
 
         if email is None or activation_token is None:
             #  We don't have the email or activation_token, show error.
-            return trans.show_error_message( "You are using wrong activation link. Try to log-in and we will send you a new activation email.<br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller="root", action="index" )
+            return trans.show_error_message( "You are using wrong activation link. Try to log-in and we will send you a new activation email. <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller="root", action="index" )
         else:
             #  Find the user
             user = trans.sa_session.query( trans.app.model.User ).filter( trans.app.model.User.table.c.email==email ).first()
+            # If the user is active already don't try to activate            
+            if user.active == True:
+                return trans.show_ok_message( "Your account is already active. Nothing has changed. <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
             if user.activation_token == activation_token:
                 user.activation_token = None
                 user.active = True
                 trans.sa_session.add(user)
                 trans.sa_session.flush()
-                return trans.show_ok_message( "Your account has been successfully activated!<br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
+                return trans.show_ok_message( "Your account has been successfully activated! <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
             else:
                 #  Tokens don't match. Activation is denied.
-                return trans.show_error_message( "You are using wrong activation link. Try to log in and we will send you a new activation email.<br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
+                return trans.show_error_message( "You are using wrong activation link. Try to log in and we will send you a new activation email. <br><a href='%s'>Go to login page.</a>" ) % web.url_for( controller='root', action='index' )
         return
 
     def __get_user_type_form_definition( self, trans, user=None, **kwd ):
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b static/scripts/galaxy.library.js
--- /dev/null
+++ b/static/scripts/galaxy.library.js
@@ -0,0 +1,902 @@
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+// === GALAXY LIBRARY MODULE ====
+// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
+
+// global variables
+var view            = null;
+var library_router  = null;
+var responses = [];
+
+// dependencies
+define([
+    "galaxy.modal", 
+    "galaxy.masthead", 
+    "utils/galaxy.utils", 
+    "libs/toastr"], function(mod_modal, mod_masthead, mod_utils, mod_toastr) {
+
+// MMMMMMMMMMMMMMM
+// === Models ====
+// MMMMMMMMMMMMMMM
+
+    // LIBRARY
+    var Library = Backbone.Model.extend({
+      urlRoot: '/api/libraries'
+  });
+
+    // LIBRARIES
+    var Libraries = Backbone.Collection.extend({
+      url: '/api/libraries',
+      model: Library
+  });
+
+    // ITEM
+    var Item = Backbone.Model.extend({
+        urlRoot : '/api/libraries/datasets'
+    })
+
+    // FOLDER
+    var Folder = Backbone.Collection.extend({
+      model: Item
+  })
+
+    // CONTAINER for folder contents (folders, items and metadata).
+    var FolderContainer = Backbone.Model.extend({
+     defaults : {
+        folder : new Folder(),
+        full_path : "unknown",
+        urlRoot : "/api/folders/",
+        id : "unknown"
+    },
+    parse : function(obj) {
+      this.full_path = obj[0].full_path;
+          // update the inner collection
+          this.get("folder").reset(obj[1].folder_contents);
+          return obj;
+      }
+  })
+
+    // HISTORY ITEM
+    var HistoryItem = Backbone.Model.extend({
+        urlRoot : '/api/histories/'
+    });
+
+    // HISTORY
+    var GalaxyHistory = Backbone.Model.extend({
+        url : '/api/histories/'
+    });
+
+    // HISTORIES
+    var GalaxyHistories = Backbone.Collection.extend({
+        url : '/api/histories',
+        model : GalaxyHistory
+    });
+
+    //ROUTER
+    var LibraryRouter = Backbone.Router.extend({
+        routes: {
+            "" :                 "libraries",
+            "folders/:id" :      "folder_content",
+            "folders/:folder_id/download/:format" : "download"
+        }
+    });
+
+
+// MMMMMMMMMMMMMM
+// === VIEWS ====
+// MMMMMMMMMMMMMM
+
+// galaxy folder
+var FolderContentView = Backbone.View.extend({
+    // main element definition
+    el : '#center',
+    // progress percentage
+    progress: 0,
+    // progress rate per one item
+    progressStep: 1,
+    // last selected history in modal for UX
+    lastSelectedHistory: '',
+    // self modal
+    modal : null,
+    // loaded folders
+    folders : null,
+
+    // initialize
+    initialize : function(){
+        this.folders = [];
+        this.queue = jQuery.Deferred();
+        this.queue.resolve();
+    },
+
+// MMMMMMMMMMMMMMMMMM
+// === TEMPLATES ====
+// MMMMMMMMMMMMMMMMMM
+
+    // set up
+    templateFolder : function (){
+        var tmpl_array = [];
+
+        // CONTAINER
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; ">');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
+
+        // TOOLBAR
+        tmpl_array.push('<div id="library_folder_toolbar" >');
+        tmpl_array.push('   <button title="Create New Folder" id="toolbtn_create_folder" class="btn btn-primary" type="button"><span class="fa fa-plus"></span><span class="fa fa-folder-close"></span> folder</button>');
+        tmpl_array.push('   <button id="toolbtn_bulk_import" class="btn btn-primary" style="display: none; margin-left: 0.5em;" type="button"><span class="fa fa-external-link"></span> to history</button>');
+
+        tmpl_array.push('   <div id="toolbtn_dl" class="btn-group" style="margin-left: 0.5em; display: none; ">');
+        tmpl_array.push('       <button id="drop_toggle" type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown">');
+        tmpl_array.push('       <span class="fa fa-download"></span> download <span class="caret"></span>');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       <ul class="dropdown-menu" role="menu">');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tgz">.tar.gz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/tbz">.tar.bz</a></li>');
+        tmpl_array.push('          <li><a href="#/folders/<%= id %>/download/zip">.zip</a></li>');
+        tmpl_array.push('       </ul>');
+        tmpl_array.push('   </div>');
+
+        tmpl_array.push('</div>');
+
+        // BREADCRUMBS
+        tmpl_array.push('<div class="library_breadcrumb">');
+        tmpl_array.push('<a title="Return to the list of libraries" href="#">Libraries</a><b>|</b> ');
+        tmpl_array.push('<% _.each(path, function(path_item) { %>');
+        tmpl_array.push('<% if (path_item[0] != id) { %>');
+        tmpl_array.push('<a title="Return to this folder" href="#/folders/<%- path_item[0] %>"><%- path_item[1] %></a><b>|</b> ');
+        tmpl_array.push('<% } else { %>');
+        tmpl_array.push('<span title="You are in this folder"><%- path_item[1] %></span>');
+        tmpl_array.push('<% } %>');
+        tmpl_array.push('<% }); %>');
+        tmpl_array.push('</div>');
+
+        // FOLDER CONTENT
+        tmpl_array.push('<table id="folder_table" class="table table-condensed">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('       <th style="text-align: center; width: 20px; "><input id="select-all-checkboxes" style="margin: 0;" type="checkbox"></th>');
+        tmpl_array.push('       <th class="button_heading">view</th>');
+        tmpl_array.push('       <th>name</th>');
+        tmpl_array.push('       <th>data type</th>');
+        tmpl_array.push('       <th>size</th>');
+        tmpl_array.push('       <th>date</th>');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       <td><button title="Go to parent folder" type="button" data-id="<%- upper_folder_id %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('       <span class="fa fa-arrow-up"></span> .. go up</td>');
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       <td></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <% _.each(items, function(content_item) { %>');
+        tmpl_array.push('       <tr class="folder_row light" id="<%- content_item.id %>">');
+        tmpl_array.push('           <% if (content_item.get("type") === "folder") { %>'); // folder
+        tmpl_array.push('               <td></td>');
+        tmpl_array.push('               <td><button title="Open this folder" type="button" data-id="<%- content_item.id %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- content_item.get("name") %>');
+        tmpl_array.push('           <% if (content_item.get("item_count") === 0) { %>'); // empty folder
+        tmpl_array.push('           <span class="muted">(empty folder)</span>');
+        tmpl_array.push('           <% } %>');
+        tmpl_array.push('           </td>');
+        tmpl_array.push('           <td>folder</td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("item_count")) %> item(s)</td>'); // size
+        tmpl_array.push('           <% } else {  %>');
+        tmpl_array.push('           <td style="text-align: center; "><input style="margin: 0;" type="checkbox"></td>');
+        tmpl_array.push('       <td>');
+        tmpl_array.push('       <button title="See details of this dataset" type="button" class="library-dataset btn btn-default btn-xs">');
+        tmpl_array.push('       <span class="fa fa-eye"></span> details');
+        tmpl_array.push('       </button>');
+        tmpl_array.push('       </td>');
+        tmpl_array.push('           <td><%- content_item.get("name") %></td>'); // dataset
+        tmpl_array.push('           <td><%= _.escape(content_item.get("data_type")) %></td>'); // data type
+        tmpl_array.push('           <td><%= _.escape(content_item.get("readable_size")) %></td>'); // size
+        tmpl_array.push('           <% } %>  ');
+        tmpl_array.push('           <td><%= _.escape(content_item.get("time_updated")) %></td>'); // time updated
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('       ');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+
+        tmpl_array.push('</div>');
+        return tmpl_array.join('');
+    },
+    templateDatasetModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<div id="dataset_info_modal">');
+        tmpl_array.push('   <table class="table table-striped table-condensed">');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row" id="id_row" data-id="<%= _.escape(item.get("ldda_id")) %>">Name</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("name")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Data type</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("data_type")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Genome build</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("genome_build")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('           <th scope="row">Size</th>');
+        tmpl_array.push('           <td><%= _.escape(size) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Date uploaded</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("date_uploaded")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Uploaded by</th>');
+        tmpl_array.push('           <td><%= _.escape(item.get("uploaded_by")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('           <tr scope="row">');
+        tmpl_array.push('           <th scope="row">Data Lines</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_data_lines")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <th scope="row">Comment Lines</th>');
+        tmpl_array.push('           <% if (item.get("metadata_comment_lines") === "") { %>'); //folder
+        tmpl_array.push('               <td scope="row"><%= _.escape(item.get("metadata_comment_lines")) %></td>');
+        tmpl_array.push('           <% } else { %>');
+        tmpl_array.push('               <td scope="row">unknown</td>');
+        tmpl_array.push('           <% } %>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Number of Columns</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_columns")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Column Types</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("metadata_column_types")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('       <tr>');
+        tmpl_array.push('           <th scope="row">Miscellaneous information</th>');
+        tmpl_array.push('           <td scope="row"><%= _.escape(item.get("misc_blurb")) %></td>');
+        tmpl_array.push('       </tr>');
+        tmpl_array.push('   </table>');
+        tmpl_array.push('   <pre class="peek">');
+        tmpl_array.push('   </pre>');
+        tmpl_array.push('</div>');
+
+        return tmpl_array.join('');
+    },
+
+    templateHistorySelectInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_combo" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_single" name="dataset_import_single" style="width:50%; margin-bottom: 1em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
+
+        return tmpl_array.join('');
+        },
+
+    templateBulkImportInModal : function(){
+        var tmpl_array = [];
+
+        tmpl_array.push('<span id="history_modal_combo_bulk" style="width:90%; margin-left: 1em; margin-right: 1em; ">');
+        tmpl_array.push('Select history: ');
+        tmpl_array.push('<select id="dataset_import_bulk" name="dataset_import_bulk" style="width:50%; margin-bottom: 1em; "> ');
+        tmpl_array.push('   <% _.each(histories, function(history) { %>'); //history select box
+        tmpl_array.push('       <option value="<%= _.escape(history.get("id")) %>"><%= _.escape(history.get("name")) %></option>');
+        tmpl_array.push('   <% }); %>');
+        tmpl_array.push('</select>');
+        tmpl_array.push('</span>');
+
+        return tmpl_array.join('');
+        },
+
+      // convert size to nice string
+      size_to_string : function (size)
+      {
+            // identify unit
+            var unit = "";
+            if (size >= 100000000000)   { size = size / 100000000000; unit = "TB"; } else
+            if (size >= 100000000)      { size = size / 100000000; unit = "GB"; } else
+            if (size >= 100000)         { size = size / 100000; unit = "MB"; } else
+            if (size >= 100)            { size = size / 100; unit = "KB"; } else
+            { size = size * 10; unit = "b"; }
+            // return formatted string
+            return (Math.round(size) / 10) + unit;
+        },
+
+// MMMMMMMMMMMMMMM
+// === EVENTS ====
+// MMMMMMMMMMMMMMM
+
+      // event binding
+      events: {
+            'click #select-all-checkboxes' : 'selectAll',
+            'click .folder_row' : 'selectClickedRow',
+            'click #toolbtn_bulk_import' : 'modalBulkImport',
+            'click #toolbtn_dl' : 'bulkDownload',
+            'click .library-dataset' : 'showDatasetDetails',
+            'click #toolbtn_create_folder' : 'createFolderModal',
+            'click .btn_open_folder' : 'navigateToFolder'
+        },
+
+      //render the folder view
+      render: function (options) {
+        //hack to show scrollbars
+        $("#center").css('overflow','auto');
+
+        view = this;
+        var that = this;
+
+        var folderContainer = new FolderContainer({id: options.id});
+        folderContainer.url = folderContainer.attributes.urlRoot + options.id + '/contents';
+
+        folderContainer.fetch({
+          success: function (container) {
+
+            // prepare nice size strings
+              for (var i = 0; i < folderContainer.attributes.folder.models.length; i++) {
+                  var model = folderContainer.attributes.folder.models[i]
+                  if (model.get('type') === 'file'){
+                    model.set('readable_size', that.size_to_string(model.get('file_size')))
+                  }
+              };
+
+              // find the upper id
+              var path = folderContainer.full_path;
+              var upper_folder_id;
+              if (path.length === 1){ // library is above us
+                upper_folder_id = 0;
+              } else {
+                upper_folder_id = path[path.length-2][0];
+              }
+
+              var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id, upper_folder_id: upper_folder_id });
+              // var template = _.template(that.templateFolder(), { path: folderContainer.full_path, items: folderContainer.attributes.folder.models, id: options.id });
+              that.$el.html(template);
+
+            }
+          })
+        },
+
+      // handles the click on 'open' and 'upper' folder icons
+      navigateToFolder : function(event){
+        var folder_id = $(event.target).attr('data-id');
+        if (typeof folder_id === 'undefined') {
+            return false;
+        } else if (folder_id === '0'){
+            library_router.navigate('#', {trigger: true, replace: true});
+        } else {
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+        }
+      },
+
+      //show modal with current dataset info
+      showDatasetDetails : function(event){
+        // prevent default
+        event.preventDefault();
+
+//TODO check whether we already have the data
+
+        //load the ID of the row
+        var id = $(event.target).parent().parent().attr('id');
+
+        //create new item
+        var item = new Item();
+        var histories = new GalaxyHistories();
+        item.id = id;
+        var self = this;
+
+        //fetch the dataset info
+        item.fetch({
+          success: function (item) {
+// TODO can render here already
+                //fetch user histories for import purposes
+                histories.fetch({
+                    success: function (histories){self.renderModalAfterFetch(item, histories)}
+                });
+            }
+        });
+    },
+
+      // show the current dataset in a modal
+      renderModalAfterFetch : function(item, histories){
+        var size = this.size_to_string(item.get('file_size'));
+        var template = _.template(this.templateDatasetModal(), { item : item, size : size });
+        this.modal = null;
+            // make modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal({
+                title   : 'Dataset Details',
+                body    : template,
+                buttons : {
+                    'Import' : function() { self.importCurrentIntoHistory() },
+                    'Download' : function() { self.downloadCurrent() },
+                    'Close'  : function() { self.modal.hide(); $('.modal').remove(); self.modal = null; } // TODO refill nicely modal with data
+                }
+            });
+            $(".peek").html(item.get("peek"));
+            var history_footer_tmpl = _.template(this.templateHistorySelectInModal(), {histories : histories.models});
+            $(this.modal.elMain).find('.buttons').prepend(history_footer_tmpl);
+
+            // preset last selected history if we know it
+            if (self.lastSelectedHistory.length > 0) {
+                $(this.modal.elMain).find('#dataset_import_single').val(self.lastSelectedHistory);
+            }
+
+            // show the prepared modal
+            this.modal.show();
+        },
+
+        // download dataset shown currently in modal
+        downloadCurrent : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
+
+            var library_dataset_id = [];
+            library_dataset_id.push($('#id_row').attr('data-id'));
+            var url = '/api/libraries/datasets/download/uncompressed';
+            var data = {'ldda_ids' : library_dataset_id};
+
+            // we assume the view is existent
+            folderContentView.processDownload(url, data);
+            this.modal.enableButton('Import');
+            this.modal.enableButton('Download');
+        },
+
+        // import dataset shown currently in modal into selected history
+        importCurrentIntoHistory : function(){
+            //disable the buttons
+            this.modal.disableButton('Import');
+            this.modal.disableButton('Download');
+
+            var history_id = $(this.modal.elMain).find('select[name=dataset_import_single] option:selected').val();
+            this.lastSelectedHistory = history_id; //save selected history for further use
+
+            var library_dataset_id = $('#id_row').attr('data-id');
+            var historyItem = new HistoryItem();
+            var self = this;
+            historyItem.url = historyItem.urlRoot + history_id + '/contents';
+
+            // save the dataset into selected history
+            historyItem.save({ content : library_dataset_id, source : 'library' }, { success : function(){
+                mod_toastr.success('Dataset imported');
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');
+            }, error : function(){
+                mod_toastr.error('An error occured! Dataset not imported. Please try again.')
+                //enable the buttons
+                self.modal.enableButton('Import');
+                self.modal.enableButton('Download');
+            }
+        });
+        },
+
+        // select all datasets
+        selectAll : function (event) {
+             var selected = event.target.checked;
+             that = this;
+             // Iterate each checkbox
+             $(':checkbox').each(function () { 
+                this.checked = selected; 
+                $row = $(this.parentElement.parentElement);
+                // Change color of selected/unselected
+                (selected) ? that.makeDarkRow($row) : that.makeWhiteRow($row);
+            });
+             // Show the tools in menu
+             this.checkTools();
+         },
+
+         // Check checkbox on row itself or row checkbox click
+         selectClickedRow : function (event) {
+            var checkbox = '';
+            var $row;
+            var source;
+            if (event.target.localName === 'input'){
+                checkbox = event.target;
+                $row = $(event.target.parentElement.parentElement);
+                source = 'input';
+            } else if (event.target.localName === 'td') {
+                checkbox = $("#" + event.target.parentElement.id).find(':checkbox')[0];
+                $row = $(event.target.parentElement);
+                source = 'td';
+            }
+            if (checkbox === '') {event.stopPropagation(); return;} // button in row was clicked
+
+                if (checkbox.checked){
+                    if (source==='td'){
+                        checkbox.checked = '';
+                        this.makeWhiteRow($row);
+                    } else if (source==='input') {
+                        this.makeDarkRow($row);
+                    }
+                } else {
+                    if (source==='td'){
+                        checkbox.checked = 'selected';
+                        this.makeDarkRow($row);
+                    } else if (source==='input') {
+                        this.makeWhiteRow($row);
+                    }
+                }
+                this.checkTools();
+        },
+
+        makeDarkRow: function($row){
+            $row.removeClass('light');
+            $row.find('a').removeClass('light');
+            $row.addClass('dark');
+            $row.find('a').addClass('dark');
+        },
+
+        makeWhiteRow: function($row){
+            $row.removeClass('dark');
+            $row.find('a').removeClass('dark');
+            $row.addClass('light');
+            $row.find('a').addClass('light');
+        },
+
+        // show toolbar in case something is selected
+        checkTools : function(){
+            var checkedValues = $('#folder_table').find(':checked');
+            if(checkedValues.length > 0){
+                $('#toolbtn_bulk_import').show();
+                $('#toolbtn_dl').show();
+            } else {
+                $('#toolbtn_bulk_import').hide();
+                $('#toolbtn_dl').hide();
+            }
+
+        },
+
+        // show bulk import modal
+        modalBulkImport : function(){
+            var self = this;
+            // fetch histories
+            var histories = new GalaxyHistories();
+            histories.fetch({
+                    success: function (histories){
+                        // make modal
+                        var history_modal_tmpl =  _.template(self.templateBulkImportInModal(), {histories : histories.models});
+                        self.modal = new mod_modal.GalaxyModal({
+                            title   : 'Import into History',
+                            body    : history_modal_tmpl,
+                            buttons : {
+                                'Import' : function() {self.importAllIntoHistory()},
+                                'Close'  : function() {self.modal.hide(); $('.modal').remove(); self.modal = null;}
+                            }
+                        });
+                        // show the prepared modal
+                        self.modal.show();
+                    }
+                });
+        },
+
+        // import all selected datasets into history
+        importAllIntoHistory : function (){
+            //disable the button
+            this.modal.disableButton('Import');
+
+            var history_id = $("select[name=dataset_import_bulk] option:selected").val();
+            var history_name = $("select[name=dataset_import_bulk] option:selected").text();
+
+            var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+            var progress_bar_tmpl = _.template(this.templateProgressBar(), { history_name : history_name });
+            $(this.modal.elMain).find('.modal-body').html(progress_bar_tmpl);
+
+            // init the progress bar
+            var progressStep = 100 / dataset_ids.length;
+            this.initProgress(progressStep);
+
+            // prepare the dataset objects to be imported
+            var datasets_to_import = [];
+            for (var i = dataset_ids.length - 1; i >= 0; i--) {
+                library_dataset_id = dataset_ids[i];
+                var historyItem = new HistoryItem();
+                var self = this;
+                historyItem.url = historyItem.urlRoot + history_id + '/contents';
+                historyItem.content = library_dataset_id;
+                historyItem.source = 'library';
+                datasets_to_import.push(historyItem);
+            };
+
+            // call the recursive function to call ajax one after each other
+            this.chainCall(datasets_to_import);
+        },
+
+        chainCall: function(history_item_set){
+            var self = this;
+            var popped_item = history_item_set.pop();
+            if (typeof popped_item === "undefined") {
+                mod_toastr.success('All datasets imported');
+                this.modal.hide();
+                // enable button again
+                self.modal.enableButton('Import');
+                return
+            }
+                var promise = $.when(popped_item.save({content: popped_item.content, source: popped_item.source})).done(function(a1){
+                        self.updateProgress();
+                        responses.push(a1);
+                        self.chainCall(history_item_set);
+                    });
+        },
+
+        initProgress: function(progressStep){
+            this.progress = 0;
+            this.progressStep = progressStep;
+        },
+        updateProgress: function(){
+            this.progress += this.progressStep;
+            $('.progress-bar').width(Math.round(this.progress) + '%');
+            txt_representation = Math.round(this.progress) + '% Complete';
+            $('.completion_span').text(txt_representation);
+        },
+
+      // progress bar
+      templateProgressBar : function (){
+        var tmpl_array = [];
+
+        tmpl_array.push('<div class="import_text">');
+        tmpl_array.push('Importing selected datasets to history <b><%= _.escape(history_name) %></b>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('<div class="progress">');
+        tmpl_array.push('   <div class="progress-bar" role="progressbar" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100" style="width: 00%;">');
+        tmpl_array.push('       <span class="completion_span">0% Complete</span>');
+        tmpl_array.push('   </div>');
+        tmpl_array.push('</div>');
+        tmpl_array.push('');
+
+        return tmpl_array.join('');
+      },
+
+      // download selected datasets
+      download : function(folder_id, format){
+        var dataset_ids = [];
+            $('#folder_table').find(':checked').each(function(){
+                if (this.parentElement.parentElement.id != '') {
+                    dataset_ids.push(this.parentElement.parentElement.id);
+                }
+            });
+
+        var url = '/api/libraries/datasets/download/' + format;
+        var data = {'ldda_ids' : dataset_ids};
+        this.processDownload(url, data, 'get');
+      },
+
+      // create hidden form and submit through POST to initialize download
+      processDownload: function(url, data, method){
+        //url and data options required
+        if( url && data ){
+                //data can be string of parameters or array/object
+                data = typeof data == 'string' ? data : $.param(data);
+                //split params into form inputs
+                var inputs = '';
+                $.each(data.split('&'), function(){
+                        var pair = this.split('=');
+                        inputs+='<input type="hidden" name="'+ pair[0] +'" value="'+ pair[1] +'" />';
+                });
+                //send request
+                $('<form action="'+ url +'" method="'+ (method||'post') +'">'+inputs+'</form>')
+                .appendTo('body').submit().remove();
+                mod_toastr.info('Your download will begin soon');
+        };
+      },
+
+      // shows modal for creating folder
+      createFolderModal: function(){
+        mod_toastr.info('This will create folder...in the future');
+      }
+
+    });
+
+// galaxy library view
+var GalaxyLibraryview = Backbone.View.extend({
+    el: '#center',
+
+    events: {
+        'click #create_new_library_btn' : 'show_library_modal'
+    },
+
+    // initialize
+    initialize : function(){
+    },
+
+    // template
+    template_library_list : function (){
+        tmpl_array = [];
+        tmpl_array.push('<div id="library_container" style="width: 90%; margin: auto; margin-top: 2em; overflow: auto !important; ">');
+
+        tmpl_array.push('');
+        tmpl_array.push('<h3>New Data Libraries. This is work in progress. Report problems & ideas to <a href="mailto:marten@bx.psu.edu?Subject=DataLibraries_Feedback" target="_blank">Marten</a>.</h3>');
+        tmpl_array.push('<a href="" id="create_new_library_btn" class="btn btn-primary file ">New Library</a>');
+        tmpl_array.push('<table class="table table-condensed">');
+        tmpl_array.push('   <thead>');
+        tmpl_array.push('     <th class="button_heading"></th>');
+        tmpl_array.push('     <th>name</th>');
+        tmpl_array.push('     <th>description</th>');
+        tmpl_array.push('     <th>synopsis</th> ');
+        tmpl_array.push('     <th>model type</th> ');
+        tmpl_array.push('   </thead>');
+        tmpl_array.push('   <tbody>');
+        tmpl_array.push('       <% _.each(libraries, function(library) { %>');
+        tmpl_array.push('           <tr>');
+        tmpl_array.push('               <td><button title="Open this library" type="button" data-id="<%- library.get("root_folder_id") %>" class="btn_open_folder btn btn-default btn-xs">');
+        tmpl_array.push('               <span class="fa fa-folder-open"></span> browse</td>');
+        tmpl_array.push('               <td><%- library.get("name") %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("description")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("synopsis")) %></td>');
+        tmpl_array.push('               <td><%= _.escape(library.get("model_class")) %></td>');
+        tmpl_array.push('           </tr>');
+        tmpl_array.push('       <% }); %>');
+        tmpl_array.push('   </tbody>');
+        tmpl_array.push('</table>');
+
+        tmpl_array.push('</div>');
+        return tmpl_array.join('');
+    },
+
+    // render
+    render: function () {
+        //hack to show scrollbars due to #center element inheritance
+        $("#center").css('overflow','auto');
+        
+        var that = this;
+        libraries = new Libraries();
+
+        libraries.fetch({
+          success: function (libraries) {
+            var template = _.template(that.template_library_list(), { libraries : libraries.models });
+            that.$el.html(template);
+          },
+          error: function(model, response){
+            if (response.statusCode().status === 403){
+                mod_toastr.error('Please log in first. Redirecting to login page in 3s.');   
+                setTimeout(that.redirectToLogin, 3000);
+            } else {
+                mod_toastr.error('An error occured. Please try again.');
+            }
+          }
+    })
+    },
+
+    redirectToHome: function(){
+        window.location = '../';
+    },    
+    redirectToLogin: function(){
+        window.location = '/user/login';
+    },
+
+    // own modal
+    modal : null,
+
+    // show/hide create library modal
+    show_library_modal : function (event){
+        event.preventDefault();
+        event.stopPropagation();
+
+        // create modal
+            var self = this;
+            this.modal = new mod_modal.GalaxyModal(
+            {
+                title   : 'Create New Library',
+                body    : this.template_new_library(),
+                buttons : {
+                    'Create' : function() {self.create_new_library_event()},
+                    'Close'  : function() {self.modal.hide()}
+                }
+            });
+
+        // show prepared modal
+        this.modal.show();
+    },
+
+    // create the new library from modal
+    create_new_library_event: function(){
+        var libraryDetails = this.serialize_new_library();
+        if (this.validate_new_library(libraryDetails)){
+            var library = new Library();
+            var self = this;
+            library.save(libraryDetails, {
+              success: function (library) {
+                self.modal.hide();
+                self.clear_library_modal();
+                self.render();
+                mod_toastr.success('Library created');
+              },
+              error: function(){
+                mod_toastr.error('An error occured :(');
+              }
+            });
+        } else {
+            mod_toastr.error('Library\'s name is missing');
+        }
+        return false;
+    },
+
+    // clear the library modal once saved
+    clear_library_modal : function(){
+        $("input[name='Name']").val('');
+        $("input[name='Description']").val('');
+        $("input[name='Synopsis']").val('');
+    },
+
+    // serialize data from the form
+    serialize_new_library : function(){
+        return {
+            name: $("input[name='Name']").val(),
+            description: $("input[name='Description']").val(),
+            synopsis: $("input[name='Synopsis']").val()
+        };
+    },
+
+    // validate new library info
+    validate_new_library: function(libraryDetails){
+        return libraryDetails.name !== '';
+    },
+
+    // template for new library modal
+    template_new_library: function(){
+        tmpl_array = [];
+
+        tmpl_array.push('<div id="new_library_modal">');
+        tmpl_array.push('<form>');
+        tmpl_array.push('<input type="text" name="Name" value="" placeholder="Name">');
+        tmpl_array.push('<input type="text" name="Description" value="" placeholder="Description">');
+        tmpl_array.push('<input type="text" name="Synopsis" value="" placeholder="Synopsis">');
+        tmpl_array.push('</form>');
+        tmpl_array.push('</div>');
+
+        return tmpl_array.join('');
+    }
+});
+
+// galaxy library wrapper View
+var GalaxyLibrary = Backbone.View.extend({
+    folderContentView : null,
+    galaxyLibraryview : null,
+    initialize : function(){
+
+        folderContentView = new FolderContentView();
+        galaxyLibraryview = new GalaxyLibraryview();
+        library_router = new LibraryRouter();
+
+        library_router.on('route:libraries', function() {
+          // render libraries list
+          galaxyLibraryview.render();
+        });
+
+        library_router.on('route:folder_content', function(id) {
+          // render folder's contents
+          folderContentView.render({id: id});
+        });
+
+       library_router.on('route:download', function(folder_id, format) {
+          if ($('#center').find(':checked').length === 0) { 
+            // this happens rarely when there is a server/data error and client gets an actual response instead of an attachment
+            // we don't know what was selected so we can't download again, we redirect to the folder provided
+            library_router.navigate('folders/' + folder_id, {trigger: true, replace: true});
+          } else {
+            // send download stream
+            folderContentView.download(folder_id, format);
+            library_router.navigate('folders/' + folder_id, {trigger: false, replace: true});
+          }
+        });
+
+Backbone.history.start();
+
+return this
+}
+});
+
+// return
+return {
+    GalaxyApp: GalaxyLibrary
+};
+
+});
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b static/scripts/galaxy.menu.js
--- a/static/scripts/galaxy.menu.js
+++ b/static/scripts/galaxy.menu.js
@@ -57,7 +57,12 @@
 
         tab_shared.add({
             title   : "Data Libraries",
-            content : "library/index",
+            content : "library/index"
+        });
+
+        tab_shared.add({
+            title   : "New Libraries",
+            content : "library/list",
             divider : true
         });
 
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b static/scripts/galaxy.modal.js
--- a/static/scripts/galaxy.modal.js
+++ b/static/scripts/galaxy.modal.js
@@ -1,6 +1,3 @@
-/*
-    galaxy modal
-*/
 
 // dependencies
 define([], function() {
@@ -25,8 +22,50 @@
     
     // initialize
     initialize : function(options) {
+        self = this;
         if (options)
             this.create(options);
+
+        // Bind the hiding events
+        this.bindEvents(event, self);
+    },
+
+    // bind the click-to-hide function
+    bindEvents: function(event, that) {
+        // bind the ESC key to hide() function
+        $(document).on('keyup', function(event){
+            if (event.keyCode == 27) { self.hide(); }
+        })
+        // bind the 'click anywhere' to hide() function...
+        $('html').on('click', function(event){
+            that.hide();
+        })
+        // ...but don't hide if the click is on modal content
+        $('.modal-content').on('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+    // unbind the click-to-hide function
+    unbindEvents: function(event, that){
+        // bind the ESC key to hide() function
+        $(document).off('keyup', function(event){
+            if (event.keyCode == 27) { that.hide(); }
+        })
+        // unbind the 'click anywhere' to hide() function...
+        $('html').off('click', function(event){
+            that.hide();
+        })
+        $('.modal-content').off('click', function(event){
+            event.stopPropagation();
+        })
+    },
+
+    // destroy
+    destroy : function(){
+        this.hide();
+        this.unbindEvents();
+        $('.modal').remove();
     },
 
     // adds and displays a new frame/window
@@ -63,6 +102,7 @@
         
         // set flag
         this.visible = false;
+        this.unbindEvents();
     },
     
     // create
@@ -120,6 +160,15 @@
         this.$buttons.find('#' + String(name).toLowerCase()).prop('disabled', true);
     },
     
+    // hide buttons
+    hideButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).hide();
+    },
+    // show buttons
+    showButton: function(name) {
+        this.$buttons.find('#' + String(name).toLowerCase()).show();
+    },
+
     // returns scroll top for body element
     scrollTop: function()
     {
@@ -139,7 +188,7 @@
                             '<div class="modal-header">' +
                                 '<button type="button" class="close" style="display: none;">×</button>' +
                                 '<h4 class="title">' + title + '</h4>' +
-                            '</div>' +
+                                '</div>' +
                             '<div class="modal-body"></div>' +
                             '<div class="modal-footer">' +
                                 '<div class="buttons" style="float: right;"></div>' +
diff -r e304da0a99681c7fb8538c6861844f17c375097e -r c1963ee06cb0fc066c111b23d99a3ea49d66f90b static/scripts/libs/bootstrap.js
--- a/static/scripts/libs/bootstrap.js
+++ b/static/scripts/libs/bootstrap.js
@@ -575,3 +575,158 @@
   }
 
 }(window.jQuery);
+
+/* ========================================================================
+ * Bootstrap: dropdown.js v3.0.0
+ * http://twbs.github.com/bootstrap/javascript.html#dropdowns
+ * ========================================================================
+ * Copyright 2012 Twitter, Inc.
+ *
+ * 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.
+ * ======================================================================== */
+
+
++function ($) { "use strict";
+
+  // DROPDOWN CLASS DEFINITION
+  // =========================
+
+  var backdrop = '.dropdown-backdrop'
+  var toggle   = '[data-toggle=dropdown]'
+  var Dropdown = function (element) {
+    var $el = $(element).on('click.bs.dropdown', this.toggle)
+  }
+
+  Dropdown.prototype.toggle = function (e) {
+    var $this = $(this)
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    clearMenus()
+
+    if (!isActive) {
+      if ('ontouchstart' in document.documentElement && !$parent.closest('.navbar-nav').length) {
+        // if mobile we we use a backdrop because click events don't delegate
+        $('<div class="dropdown-backdrop"/>').insertAfter($(this)).on('click', clearMenus)
+      }
+
+      $parent.trigger(e = $.Event('show.bs.dropdown'))
+
+      if (e.isDefaultPrevented()) return
+
+      $parent
+        .toggleClass('open')
+        .trigger('shown.bs.dropdown')
+
+      $this.focus()
+    }
+
+    return false
+  }
+
+  Dropdown.prototype.keydown = function (e) {
+    if (!/(38|40|27)/.test(e.keyCode)) return
+
+    var $this = $(this)
+
+    e.preventDefault()
+    e.stopPropagation()
+
+    if ($this.is('.disabled, :disabled')) return
+
+    var $parent  = getParent($this)
+    var isActive = $parent.hasClass('open')
+
+    if (!isActive || (isActive && e.keyCode == 27)) {
+      if (e.which == 27) $parent.find(toggle).focus()
+      return $this.click()
+    }
+
+    var $items = $('[role=menu] li:not(.divider):visible a', $parent)
+
+    if (!$items.length) return
+
+    var index = $items.index($items.filter(':focus'))
+
+    if (e.keyCode == 38 && index > 0)                 index--                        // up
+    if (e.keyCode == 40 && index < $items.length - 1) index++                        // down
+    if (!~index)                                      index=0
+
+    $items.eq(index).focus()
+  }
+
+  function clearMenus() {
+    $(backdrop).remove()
+    $(toggle).each(function (e) {
+      var $parent = getParent($(this))
+      if (!$parent.hasClass('open')) return
+      $parent.trigger(e = $.Event('hide.bs.dropdown'))
+      if (e.isDefaultPrevented()) return
+      $parent.removeClass('open').trigger('hidden.bs.dropdown')
+    })
+  }
+
+  function getParent($this) {
+    var selector = $this.attr('data-target')
+
+    if (!selector) {
+      selector = $this.attr('href')
+      selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
+    }
+
+    var $parent = selector && $(selector)
+
+    return $parent && $parent.length ? $parent : $this.parent()
+  }
+
+
+  // DROPDOWN PLUGIN DEFINITION
+  // ==========================
+
+  var old = $.fn.dropdown
+
+  $.fn.dropdown = function (option) {
+    return this.each(function () {
+      var $this = $(this)
+      var data  = $this.data('dropdown')
+
+      if (!data) $this.data('dropdown', (data = new Dropdown(this)))
+      if (typeof option == 'string') data[option].call($this)
+    })
+  }
+
+  $.fn.dropdown.Constructor = Dropdown
+
+
+  // DROPDOWN NO CONFLICT
+  // ====================
+
+  $.fn.dropdown.noConflict = function () {
+    $.fn.dropdown = old
+    return this
+  }
+
+
+  // APPLY TO STANDARD DROPDOWN ELEMENTS
+  // ===================================
+
+  $(document)
+    .on('click.bs.dropdown.data-api', clearMenus)
+    .on('click.bs.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
+    .on('click.bs.dropdown.data-api'  , toggle, Dropdown.prototype.toggle)
+    .on('keydown.bs.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown)
+
+}(window.jQuery);
\ No newline at end of file
This diff is so big that we needed to truncate the remainder.
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0
                            
                          
                          
                            
    
                          
                        
                    
                        
                            
                                
                            
                            commit/galaxy-central: guerler: Grids: Attempt to	fix problems with toolshed
                        
                        
by commits-noreply@bitbucket.org 19 Dec '13
                    by commits-noreply@bitbucket.org 19 Dec '13
19 Dec '13
                    
                        1 new commit in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/e304da0a9968/
Changeset:   e304da0a9968
User:        guerler
Date:        2013-12-19 18:29:41
Summary:     Grids: Attempt to fix problems with toolshed
Affected #:  2 files
diff -r c91f3185e8ca68a006b755a9fdcbed3174b27fd9 -r e304da0a99681c7fb8538c6861844f17c375097e static/scripts/galaxy.grids.js
--- a/static/scripts/galaxy.grids.js
+++ b/static/scripts/galaxy.grids.js
@@ -322,7 +322,6 @@
             var button = $('#grid-' + i + '-popup');
             button.off();
             var popup = new PopupMenu(button);
-            
             // load details
             var item = options['items'][i];
             for (var j in options['operations'])
@@ -655,9 +654,6 @@
                 // Initialize new grid config
                 self.init_grid($.parseJSON(response_text));
                
-                // Trigger custom event to indicate grid body has changed.
-                $('#grid-table-body').trigger('update');
-               
                 // Hide loading overlay.
                 $('.loading-elt-overlay').hide();
             },
diff -r c91f3185e8ca68a006b755a9fdcbed3174b27fd9 -r e304da0a99681c7fb8538c6861844f17c375097e static/scripts/packed/galaxy.grids.js
--- a/static/scripts/packed/galaxy.grids.js
+++ b/static/scripts/packed/galaxy.grids.js
@@ -1,1 +1,1 @@
-jQuery.ajaxSettings.traditional=true;define(["mvc/ui"],function(){var a=Backbone.Model.extend({defaults:{url_base:"",async:false,async_ops:[],categorical_filters:[],filters:{},sort_key:null,show_item_checkboxes:false,advanced_search:false,cur_page:1,num_pages:1,operation:undefined,item_ids:undefined},can_async_op:function(c){return _.indexOf(this.attributes.async_ops,c)!==-1},add_filter:function(g,h,d){if(d){var e=this.attributes.filters[g],c;if(e===null||e===undefined){c=h}else{if(typeof(e)=="string"){if(e=="All"){c=h}else{var f=[];f[0]=e;f[1]=h;c=f}}else{c=e;c.push(h)}}this.attributes.filters[g]=c}else{this.attributes.filters[g]=h}},remove_filter:function(d,g){var c=this.attributes.filters[d];if(c===null||c===undefined){return false}var f=true;if(typeof(c)==="string"){if(c=="All"){f=false}else{delete this.attributes.filters[d]}}else{var e=_.indexOf(c,g);if(e!==-1){c.splice(e,1)}else{f=false}}return f},get_url_data:function(){var c={async:this.attributes.async,sort:this.attributes.sort_key,page:this.attributes.cur_page,show_item_checkboxes:this.attributes.show_item_checkboxes,advanced_search:this.attributes.advanced_search};if(this.attributes.operation){c.operation=this.attributes.operation}if(this.attributes.item_ids){c.id=this.attributes.item_ids}var d=this;_.each(_.pairs(d.attributes.filters),function(e){c["f-"+e[0]]=e[1]});return c},get_url:function(c){return this.get("url_base")+"?"+$.param(this.get_url_data())+"&"+$.param(c)}});var b=Backbone.View.extend({grid:null,initialize:function(c){this.init_grid(c);this.init_grid_controls();$("input[type=text]").each(function(){$(this).click(function(){$(this).select()}).keyup(function(){$(this).css("font-style","normal")})})},init_grid:function(e){this.grid=new a(e);var d=this.grid.attributes;var c=this.grid.get("url_base");c=c.replace(/^.*\/\/[^\/]+/,"");this.grid.set("url_base",c);$("#grid-table-body").html(this.template_body(d));$("#grid-table-footer").html(this.template_footer(d));if(d.message){$("#grid-message").html(this.template_message(d));setTimeout(function(){$("#grid-message").html("")},5000)}this.init_grid_elements()},init_grid_controls:function(){$(".submit-image").each(function(){$(this).mousedown(function(){$(this).addClass("gray-background")});$(this).mouseup(function(){$(this).removeClass("gray-background")})});var c=this;$(".sort-link").each(function(){$(this).click(function(){c.set_sort_condition($(this).attr("sort_key"));return false})});$(".categorical-filter > a").each(function(){$(this).click(function(){c.set_categorical_filter($(this).attr("filter_key"),$(this).attr("filter_val"));return false})});$(".text-filter-form").each(function(){$(this).submit(function(){var g=$(this).attr("column_key");var f=$("#input-"+g+"-filter");var h=f.val();f.val("");c.add_filter_condition(g,h);return false})});var d=$("#input-tags-filter");if(d.length){d.autocomplete(this.grid.history_tag_autocomplete_url,{selectFirst:false,autoFill:false,highlight:false,mustMatch:false})}var e=$("#input-name-filter");if(e.length){e.autocomplete(this.grid.history_name_autocomplete_url,{selectFirst:false,autoFill:false,highlight:false,mustMatch:false})}$(".advanced-search-toggle").each(function(){$(this).click(function(){$("#standard-search").slideToggle("fast");$("#advanced-search").slideToggle("fast");return false})})},init_grid_elements:function(){$(".grid").each(function(){var r=$(this).find("input.grid-row-select-checkbox");var q=$(this).find("span.grid-selected-count");var s=function(){q.text($(r).filter(":checked").length)};$(r).each(function(){$(this).change(s)});s()});if($(".community_rating_star").length!==0){$(".community_rating_star").rating({})}var p=this.grid.attributes;var o=this;$(".page-link > a").each(function(){$(this).click(function(){o.set_page($(this).attr("page_num"));return false})});$(".use-inbound").each(function(){$(this).click(function(q){o.execute({href:$(this).attr("href"),inbound:true});return false})});$(".use-outbound").each(function(){$(this).click(function(q){o.execute({href:$(this).attr("href")});return false})});for(var h in p.items){var k=$("#grid-"+h+"-popup");k.off();var d=new PopupMenu(k);var n=p.items[h];for(var g in p.operations){var e=p.operations[g];var l=e.label;var c=n.operation_config[l];var f=n.encode_id;if(c.allowed&&e.allow_popup){var m={html:e.label,href:c.url_args,target:c.target,confirmation_text:e.confirm,inbound:e.inbound};m.func=function(s){s.preventDefault();var r=$(s.target).html();var q=this.findItemByHtml(r);o.execute(q)};d.addItem(m)}}}},add_filter_condition:function(e,g){if(g===""){return false}this.grid.add_filter(e,g,true);var f=$("<span>"+g+"<a href='javascript:void(0);'><span class='delete-search-icon' /></span></a>");f.addClass("text-filter-val");var d=this;f.click(function(){d.grid.remove_filter(e,g);$(this).remove();d.go_page_one();d.execute()});var c=$("#"+e+"-filtering-criteria");c.append(f);this.go_page_one();this.execute()},set_sort_condition:function(h){var g=this.grid.get("sort_key");var f=h;if(g.indexOf(h)!==-1){if(g.substring(0,1)!=="-"){f="-"+h}else{}}$(".sort-arrow").remove();var e=(f.substring(0,1)=="-")?"↑":"↓";var c=$("<span>"+e+"</span>").addClass("sort-arrow");var d=$("#"+h+"-header");d.append(c);this.grid.set("sort_key",f);this.go_page_one();this.execute()},set_categorical_filter:function(e,g){var d=this.grid.get("categorical_filters")[e],f=this.grid.get("filters")[e];var c=this;$("."+e+"-filter").each(function(){var m=$.trim($(this).text());var k=d[m];var l=k[e];if(l==g){$(this).empty();$(this).addClass("current-filter");$(this).append(m)}else{if(l==f){$(this).empty();var h=$("<a href='#'>"+m+"</a>");h.click(function(){c.set_categorical_filter(e,l)});$(this).removeClass("current-filter");$(this).append(h)}}});this.grid.add_filter(e,g);this.go_page_one();this.execute()},set_page:function(c){var d=this;$(".page-link").each(function(){var k=$(this).attr("id"),g=parseInt(k.split("-")[2],10),e=d.grid.get("cur_page"),h;if(g===c){h=$(this).children().text();$(this).empty();$(this).addClass("inactive-link");$(this).text(h)}else{if(g===e){h=$(this).text();$(this).empty();$(this).removeClass("inactive-link");var f=$("<a href='#'>"+h+"</a>");f.click(function(){d.set_page(g)});$(this).append(f)}}});if(c==="all"){this.grid.set("cur_page",c)}else{this.grid.set("cur_page",parseInt(c,10))}this.execute()},submit_operation:function(f,g){var e=$('input[name="id"]:checked').length;if(!e>0){return false}var d=$(f).val();var c=[];$("input[name=id]:checked").each(function(){c.push($(this).val())});this.execute({operation:d,id:c,confirmation_text:g});return true},execute:function(n){var f=null;var e=null;var g=null;var c=null;var m=null;if(n){e=n.href;g=n.operation;f=n.id;c=n.confirmation_text;m=n.inbound;if(e!==undefined&&e.indexOf("operation=")!=-1){var l=e.split("?");if(l.length>1){var k=l[1];var d=k.split("&");for(var h=0;h<d.length;h++){if(d[h].indexOf("operation")!=-1){g=d[h].split("=")[1];g=g.replace(/\+/g," ")}else{if(d[h].indexOf("id")!=-1){f=d[h].split("=")[1]}}}}}}if(g&&f){if(c&&c!=""&&c!="None"&&c!="null"){if(!confirm(c)){return false}}g=g.toLowerCase();this.grid.set({operation:g,item_ids:f});if(this.grid.can_async_op(g)){this.update_grid()}else{this.go_to(m,"")}return false}if(e){this.go_to(m,e);return false}if(this.grid.get("async")){this.update_grid()}else{this.go_to(m,"")}return false},go_to:function(f,d){var e=this.grid.get("async");this.grid.set("async",false);advanced_search=$("#advanced-search").is(":visible");this.grid.set("advanced_search",advanced_search);if(!d){d=this.grid.get("url_base")+"?"+$.param(this.grid.get_url_data())}this.grid.set({operation:undefined,item_ids:undefined,async:e});if(f){var c=$(".grid-header").closest(".inbound");if(c.length!==0){c.load(d);return}}window.location=d},update_grid:function(){var d=(this.grid.get("operation")?"POST":"GET");$(".loading-elt-overlay").show();var c=this;$.ajax({type:d,url:c.grid.get("url_base"),data:c.grid.get_url_data(),error:function(e){alert("Grid refresh failed")},success:function(e){c.init_grid($.parseJSON(e));$("#grid-table-body").trigger("update");$(".loading-elt-overlay").hide()},complete:function(){c.grid.set({operation:undefined,item_ids:undefined})}})},check_all_items:function(){var c=document.getElementById("check_all"),d=document.getElementsByTagName("input"),f=0,e;if(c.checked===true){for(e=0;e<d.length;e++){if(d[e].name.indexOf("id")!==-1){d[e].checked=true;f++}}}else{for(e=0;e<d.length;e++){if(d[e].name.indexOf("id")!==-1){d[e].checked=false}}}this.init_grid_elements()},go_page_one:function(){var c=this.grid.get("cur_page");if(c!==null&&c!==undefined&&c!=="all"){this.grid.set("cur_page",1)}},template_body:function(s){var l="";var t=0;var g=s.items.length;if(g==0){l+='<tr><td colspan="100"><em>No Items</em></td></tr>';t=1}for(i in s.items){var q=s.items[i];var c=q.encode_id;var h="grid-"+i+"-popup";l+="<tr ";if(s.current_item_id==q.id){l+='class="current"'}l+=">";if(s.show_item_checkboxes){l+='<td style="width: 1.5em;"><input type="checkbox" name="id" value="'+c+'" id="'+c+'" class="grid-row-select-checkbox" /></td>'}for(j in s.columns){var f=s.columns[j];if(f.visible){var e="";if(f.nowrap){e='style="white-space:nowrap;"'}var r=q.column_config[f.label];var k=r.link;var m=r.value;var p=r.inbound;if(jQuery.type(m)==="string"){m=m.replace(/\/\//g,"/")}var d="";var o="";if(f.attach_popup){d="grid-"+i+"-popup";o="menubutton";if(k!=""){o+=" split"}o+=" popup"}l+="<td "+e+">";if(k){if(s.operations.length!=0){l+='<div id="'+d+'" class="'+o+'" style="float: left;">'}var n="";if(p){n="use-inbound"}else{n="use-outbound"}l+='<a class="label '+n+'" href="'+k+'" onclick="return false;">'+m+"</a>";if(s.operations.length!=0){l+="</div>"}}else{l+='<div id="'+d+'" class="'+o+'"><label id="'+f.label_id_prefix+c+'" for="'+c+'">'+m+"</label></div>"}l+="</td>"}}l+="</tr>";t++}return l},template_footer:function(q){var m="";if(q.use_paging&&q.num_pages>1){var o=q.num_page_links;var c=q.cur_page_num;var p=q.num_pages;var l=o/2;var k=c-l;var g=0;if(k==0){k=1;g=l-(c-k)}var f=l+g;var e=c+f;if(e<=p){max_offset=0}else{e=p;max_offset=f-(e+1-c)}if(max_offset!=0){k-=max_offset;if(k<1){k=1}}m+='<tr id="page-links-row">';if(q.show_item_checkboxes){m+="<td></td>"}m+='<td colspan="100"><span id="page-link-container">Page:';if(k>1){m+='<span class="page-link" id="page-link-1"><a href="'+this.grid.get_url({page:n})+'" page_num="1" onclick="return false;">1</a></span> ...'}for(var n=k;n<e+1;n++){if(n==q.cur_page_num){m+='<span class="page-link inactive-link" id="page-link-'+n+'">'+n+"</span>"}else{m+='<span class="page-link" id="page-link-'+n+'"><a href="'+this.grid.get_url({page:n})+'" onclick="return false;" page_num="'+n+'">'+n+"</a></span>"}}if(e<p){m+='...<span class="page-link" id="page-link-'+p+'"><a href="'+this.grid.get_url({page:p})+'" onclick="return false;" page_num="'+p+'">'+p+"</a></span>"}m+="</span>";m+='<span class="page-link" id="show-all-link-span"> | <a href="'+this.grid.get_url({page:"all"})+'" onclick="return false;" page_num="all">Show All</a></span></td></tr>'}if(q.show_item_checkboxes){m+='<tr><input type="hidden" id="operation" name="operation" value=""><td></td><td colspan="100">For <span class="grid-selected-count"></span> selected '+q.get_class_plural+": ";for(i in q.operations){var d=q.operations[i];if(d.allow_multiple){m+='<input type="button" value="'+d.label+'" class="action-button" onclick="gridView.submit_operation(this, \''+d.confirm+"')\"> "}}m+="</td></tr>"}var h=false;for(i in q.operations){if(q.operations[i].global_operation){h=true;break}}if(h){m+='<tr><td colspan="100">';for(i in q.operations){var d=q.operations[i];if(d.global_operation){m+='<a class="action-button" href="'+d.global_operation+'">'+d.label+"</a>"}}m+="</td></tr>"}if(q.legend){m+='<tr><td colspan="100">'+q.legend+"</td></tr>"}return m},template_message:function(c){return'<p><div class="'+c.status+'message transient-message">'+c.message+'</div><div style="clear: both"></div></p>'}});return{Grid:a,GridView:b}});
\ No newline at end of file
+jQuery.ajaxSettings.traditional=true;define(["mvc/ui"],function(){var a=Backbone.Model.extend({defaults:{url_base:"",async:false,async_ops:[],categorical_filters:[],filters:{},sort_key:null,show_item_checkboxes:false,advanced_search:false,cur_page:1,num_pages:1,operation:undefined,item_ids:undefined},can_async_op:function(c){return _.indexOf(this.attributes.async_ops,c)!==-1},add_filter:function(g,h,d){if(d){var e=this.attributes.filters[g],c;if(e===null||e===undefined){c=h}else{if(typeof(e)=="string"){if(e=="All"){c=h}else{var f=[];f[0]=e;f[1]=h;c=f}}else{c=e;c.push(h)}}this.attributes.filters[g]=c}else{this.attributes.filters[g]=h}},remove_filter:function(d,g){var c=this.attributes.filters[d];if(c===null||c===undefined){return false}var f=true;if(typeof(c)==="string"){if(c=="All"){f=false}else{delete this.attributes.filters[d]}}else{var e=_.indexOf(c,g);if(e!==-1){c.splice(e,1)}else{f=false}}return f},get_url_data:function(){var c={async:this.attributes.async,sort:this.attributes.sort_key,page:this.attributes.cur_page,show_item_checkboxes:this.attributes.show_item_checkboxes,advanced_search:this.attributes.advanced_search};if(this.attributes.operation){c.operation=this.attributes.operation}if(this.attributes.item_ids){c.id=this.attributes.item_ids}var d=this;_.each(_.pairs(d.attributes.filters),function(e){c["f-"+e[0]]=e[1]});return c},get_url:function(c){return this.get("url_base")+"?"+$.param(this.get_url_data())+"&"+$.param(c)}});var b=Backbone.View.extend({grid:null,initialize:function(c){this.init_grid(c);this.init_grid_controls();$("input[type=text]").each(function(){$(this).click(function(){$(this).select()}).keyup(function(){$(this).css("font-style","normal")})})},init_grid:function(e){this.grid=new a(e);var d=this.grid.attributes;var c=this.grid.get("url_base");c=c.replace(/^.*\/\/[^\/]+/,"");this.grid.set("url_base",c);$("#grid-table-body").html(this.template_body(d));$("#grid-table-footer").html(this.template_footer(d));if(d.message){$("#grid-message").html(this.template_message(d));setTimeout(function(){$("#grid-message").html("")},5000)}this.init_grid_elements()},init_grid_controls:function(){$(".submit-image").each(function(){$(this).mousedown(function(){$(this).addClass("gray-background")});$(this).mouseup(function(){$(this).removeClass("gray-background")})});var c=this;$(".sort-link").each(function(){$(this).click(function(){c.set_sort_condition($(this).attr("sort_key"));return false})});$(".categorical-filter > a").each(function(){$(this).click(function(){c.set_categorical_filter($(this).attr("filter_key"),$(this).attr("filter_val"));return false})});$(".text-filter-form").each(function(){$(this).submit(function(){var g=$(this).attr("column_key");var f=$("#input-"+g+"-filter");var h=f.val();f.val("");c.add_filter_condition(g,h);return false})});var d=$("#input-tags-filter");if(d.length){d.autocomplete(this.grid.history_tag_autocomplete_url,{selectFirst:false,autoFill:false,highlight:false,mustMatch:false})}var e=$("#input-name-filter");if(e.length){e.autocomplete(this.grid.history_name_autocomplete_url,{selectFirst:false,autoFill:false,highlight:false,mustMatch:false})}$(".advanced-search-toggle").each(function(){$(this).click(function(){$("#standard-search").slideToggle("fast");$("#advanced-search").slideToggle("fast");return false})})},init_grid_elements:function(){$(".grid").each(function(){var r=$(this).find("input.grid-row-select-checkbox");var q=$(this).find("span.grid-selected-count");var s=function(){q.text($(r).filter(":checked").length)};$(r).each(function(){$(this).change(s)});s()});if($(".community_rating_star").length!==0){$(".community_rating_star").rating({})}var p=this.grid.attributes;var o=this;$(".page-link > a").each(function(){$(this).click(function(){o.set_page($(this).attr("page_num"));return false})});$(".use-inbound").each(function(){$(this).click(function(q){o.execute({href:$(this).attr("href"),inbound:true});return false})});$(".use-outbound").each(function(){$(this).click(function(q){o.execute({href:$(this).attr("href")});return false})});for(var h in p.items){var k=$("#grid-"+h+"-popup");k.off();var d=new PopupMenu(k);var n=p.items[h];for(var g in p.operations){var e=p.operations[g];var l=e.label;var c=n.operation_config[l];var f=n.encode_id;if(c.allowed&&e.allow_popup){var m={html:e.label,href:c.url_args,target:c.target,confirmation_text:e.confirm,inbound:e.inbound};m.func=function(s){s.preventDefault();var r=$(s.target).html();var q=this.findItemByHtml(r);o.execute(q)};d.addItem(m)}}}},add_filter_condition:function(e,g){if(g===""){return false}this.grid.add_filter(e,g,true);var f=$("<span>"+g+"<a href='javascript:void(0);'><span class='delete-search-icon' /></span></a>");f.addClass("text-filter-val");var d=this;f.click(function(){d.grid.remove_filter(e,g);$(this).remove();d.go_page_one();d.execute()});var c=$("#"+e+"-filtering-criteria");c.append(f);this.go_page_one();this.execute()},set_sort_condition:function(h){var g=this.grid.get("sort_key");var f=h;if(g.indexOf(h)!==-1){if(g.substring(0,1)!=="-"){f="-"+h}else{}}$(".sort-arrow").remove();var e=(f.substring(0,1)=="-")?"↑":"↓";var c=$("<span>"+e+"</span>").addClass("sort-arrow");var d=$("#"+h+"-header");d.append(c);this.grid.set("sort_key",f);this.go_page_one();this.execute()},set_categorical_filter:function(e,g){var d=this.grid.get("categorical_filters")[e],f=this.grid.get("filters")[e];var c=this;$("."+e+"-filter").each(function(){var m=$.trim($(this).text());var k=d[m];var l=k[e];if(l==g){$(this).empty();$(this).addClass("current-filter");$(this).append(m)}else{if(l==f){$(this).empty();var h=$("<a href='#'>"+m+"</a>");h.click(function(){c.set_categorical_filter(e,l)});$(this).removeClass("current-filter");$(this).append(h)}}});this.grid.add_filter(e,g);this.go_page_one();this.execute()},set_page:function(c){var d=this;$(".page-link").each(function(){var k=$(this).attr("id"),g=parseInt(k.split("-")[2],10),e=d.grid.get("cur_page"),h;if(g===c){h=$(this).children().text();$(this).empty();$(this).addClass("inactive-link");$(this).text(h)}else{if(g===e){h=$(this).text();$(this).empty();$(this).removeClass("inactive-link");var f=$("<a href='#'>"+h+"</a>");f.click(function(){d.set_page(g)});$(this).append(f)}}});if(c==="all"){this.grid.set("cur_page",c)}else{this.grid.set("cur_page",parseInt(c,10))}this.execute()},submit_operation:function(f,g){var e=$('input[name="id"]:checked').length;if(!e>0){return false}var d=$(f).val();var c=[];$("input[name=id]:checked").each(function(){c.push($(this).val())});this.execute({operation:d,id:c,confirmation_text:g});return true},execute:function(n){var f=null;var e=null;var g=null;var c=null;var m=null;if(n){e=n.href;g=n.operation;f=n.id;c=n.confirmation_text;m=n.inbound;if(e!==undefined&&e.indexOf("operation=")!=-1){var l=e.split("?");if(l.length>1){var k=l[1];var d=k.split("&");for(var h=0;h<d.length;h++){if(d[h].indexOf("operation")!=-1){g=d[h].split("=")[1];g=g.replace(/\+/g," ")}else{if(d[h].indexOf("id")!=-1){f=d[h].split("=")[1]}}}}}}if(g&&f){if(c&&c!=""&&c!="None"&&c!="null"){if(!confirm(c)){return false}}g=g.toLowerCase();this.grid.set({operation:g,item_ids:f});if(this.grid.can_async_op(g)){this.update_grid()}else{this.go_to(m,"")}return false}if(e){this.go_to(m,e);return false}if(this.grid.get("async")){this.update_grid()}else{this.go_to(m,"")}return false},go_to:function(f,d){var e=this.grid.get("async");this.grid.set("async",false);advanced_search=$("#advanced-search").is(":visible");this.grid.set("advanced_search",advanced_search);if(!d){d=this.grid.get("url_base")+"?"+$.param(this.grid.get_url_data())}this.grid.set({operation:undefined,item_ids:undefined,async:e});if(f){var c=$(".grid-header").closest(".inbound");if(c.length!==0){c.load(d);return}}window.location=d},update_grid:function(){var d=(this.grid.get("operation")?"POST":"GET");$(".loading-elt-overlay").show();var c=this;$.ajax({type:d,url:c.grid.get("url_base"),data:c.grid.get_url_data(),error:function(e){alert("Grid refresh failed")},success:function(e){c.init_grid($.parseJSON(e));$(".loading-elt-overlay").hide()},complete:function(){c.grid.set({operation:undefined,item_ids:undefined})}})},check_all_items:function(){var c=document.getElementById("check_all"),d=document.getElementsByTagName("input"),f=0,e;if(c.checked===true){for(e=0;e<d.length;e++){if(d[e].name.indexOf("id")!==-1){d[e].checked=true;f++}}}else{for(e=0;e<d.length;e++){if(d[e].name.indexOf("id")!==-1){d[e].checked=false}}}this.init_grid_elements()},go_page_one:function(){var c=this.grid.get("cur_page");if(c!==null&&c!==undefined&&c!=="all"){this.grid.set("cur_page",1)}},template_body:function(s){var l="";var t=0;var g=s.items.length;if(g==0){l+='<tr><td colspan="100"><em>No Items</em></td></tr>';t=1}for(i in s.items){var q=s.items[i];var c=q.encode_id;var h="grid-"+i+"-popup";l+="<tr ";if(s.current_item_id==q.id){l+='class="current"'}l+=">";if(s.show_item_checkboxes){l+='<td style="width: 1.5em;"><input type="checkbox" name="id" value="'+c+'" id="'+c+'" class="grid-row-select-checkbox" /></td>'}for(j in s.columns){var f=s.columns[j];if(f.visible){var e="";if(f.nowrap){e='style="white-space:nowrap;"'}var r=q.column_config[f.label];var k=r.link;var m=r.value;var p=r.inbound;if(jQuery.type(m)==="string"){m=m.replace(/\/\//g,"/")}var d="";var o="";if(f.attach_popup){d="grid-"+i+"-popup";o="menubutton";if(k!=""){o+=" split"}o+=" popup"}l+="<td "+e+">";if(k){if(s.operations.length!=0){l+='<div id="'+d+'" class="'+o+'" style="float: left;">'}var n="";if(p){n="use-inbound"}else{n="use-outbound"}l+='<a class="label '+n+'" href="'+k+'" onclick="return false;">'+m+"</a>";if(s.operations.length!=0){l+="</div>"}}else{l+='<div id="'+d+'" class="'+o+'"><label id="'+f.label_id_prefix+c+'" for="'+c+'">'+m+"</label></div>"}l+="</td>"}}l+="</tr>";t++}return l},template_footer:function(q){var m="";if(q.use_paging&&q.num_pages>1){var o=q.num_page_links;var c=q.cur_page_num;var p=q.num_pages;var l=o/2;var k=c-l;var g=0;if(k==0){k=1;g=l-(c-k)}var f=l+g;var e=c+f;if(e<=p){max_offset=0}else{e=p;max_offset=f-(e+1-c)}if(max_offset!=0){k-=max_offset;if(k<1){k=1}}m+='<tr id="page-links-row">';if(q.show_item_checkboxes){m+="<td></td>"}m+='<td colspan="100"><span id="page-link-container">Page:';if(k>1){m+='<span class="page-link" id="page-link-1"><a href="'+this.grid.get_url({page:n})+'" page_num="1" onclick="return false;">1</a></span> ...'}for(var n=k;n<e+1;n++){if(n==q.cur_page_num){m+='<span class="page-link inactive-link" id="page-link-'+n+'">'+n+"</span>"}else{m+='<span class="page-link" id="page-link-'+n+'"><a href="'+this.grid.get_url({page:n})+'" onclick="return false;" page_num="'+n+'">'+n+"</a></span>"}}if(e<p){m+='...<span class="page-link" id="page-link-'+p+'"><a href="'+this.grid.get_url({page:p})+'" onclick="return false;" page_num="'+p+'">'+p+"</a></span>"}m+="</span>";m+='<span class="page-link" id="show-all-link-span"> | <a href="'+this.grid.get_url({page:"all"})+'" onclick="return false;" page_num="all">Show All</a></span></td></tr>'}if(q.show_item_checkboxes){m+='<tr><input type="hidden" id="operation" name="operation" value=""><td></td><td colspan="100">For <span class="grid-selected-count"></span> selected '+q.get_class_plural+": ";for(i in q.operations){var d=q.operations[i];if(d.allow_multiple){m+='<input type="button" value="'+d.label+'" class="action-button" onclick="gridView.submit_operation(this, \''+d.confirm+"')\"> "}}m+="</td></tr>"}var h=false;for(i in q.operations){if(q.operations[i].global_operation){h=true;break}}if(h){m+='<tr><td colspan="100">';for(i in q.operations){var d=q.operations[i];if(d.global_operation){m+='<a class="action-button" href="'+d.global_operation+'">'+d.label+"</a>"}}m+="</td></tr>"}if(q.legend){m+='<tr><td colspan="100">'+q.legend+"</td></tr>"}return m},template_message:function(c){return'<p><div class="'+c.status+'message transient-message">'+c.message+'</div><div style="clear: both"></div></p>'}});return{Grid:a,GridView:b}});
\ No newline at end of file
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0
                            
                          
                          
                            
    
                          
                        
                    
                        
                            
                                
                            
                            commit/galaxy-central: carlfeberhard: History panel: bring error state HDA button order into line with other states
                        
                        
by commits-noreply@bitbucket.org 19 Dec '13
                    by commits-noreply@bitbucket.org 19 Dec '13
19 Dec '13
                    
                        1 new commit in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/c91f3185e8ca/
Changeset:   c91f3185e8ca
User:        carlfeberhard
Date:        2013-12-19 17:47:11
Summary:     History panel: bring error state HDA button order into line with other states
Affected #:  2 files
diff -r 4fc04317cef36f002e071d391523450a7a956232 -r c91f3185e8ca68a006b755a9fdcbed3174b27fd9 static/scripts/mvc/dataset/hda-base.js
--- a/static/scripts/mvc/dataset/hda-base.js
+++ b/static/scripts/mvc/dataset/hda-base.js
@@ -354,7 +354,7 @@
             html = '<div>' + this.model.get( 'misc_blurb' ) + '</div>' + html;
         }
         return this._render_stateBodyHelper( html,
-            this.defaultPrimaryActionButtonRenderers.concat([ this._render_downloadButton ])
+            [ this._render_downloadButton ].concat( this.defaultPrimaryActionButtonRenderers )
         );
     },
         
diff -r 4fc04317cef36f002e071d391523450a7a956232 -r c91f3185e8ca68a006b755a9fdcbed3174b27fd9 static/scripts/packed/mvc/dataset/hda-base.js
--- a/static/scripts/packed/mvc/dataset/hda-base.js
+++ b/static/scripts/packed/mvc/dataset/hda-base.js
@@ -1,1 +1,1 @@
-define(["mvc/dataset/hda-model"],function(b){var a=Backbone.View.extend(LoggableMixin).extend({tagName:"div",className:"dataset hda history-panel-hda",id:function(){return"hda-"+this.model.get("id")},fxSpeed:"fast",initialize:function(c){if(c.logger){this.logger=this.model.logger=c.logger}this.log(this+".initialize:",c);this.defaultPrimaryActionButtonRenderers=[this._render_showParamsButton];this.linkTarget=c.linkTarget||"_blank";this.selectable=c.selectable||false;this.selected=c.selected||false;this.expanded=c.expanded||false;this.draggable=c.draggable||false;this._setUpListeners()},_setUpListeners:function(){this.model.on("change",function(d,c){if(this.model.changedAttributes().state&&this.model.inReadyState()&&this.expanded&&!this.model.hasDetails()){this.model.fetch()}else{this.render()}},this)},render:function(e){e=(e===undefined)?(true):(e);var c=this;this.$el.find("[title]").tooltip("destroy");this.urls=this.model.urls();var d=$(a.templates.skeleton(this.model.toJSON()));d.find(".dataset-primary-actions").append(this._render_titleButtons());d.children(".dataset-body").replaceWith(this._render_body());this._setUpBehaviors(d);if(e){$(c).queue(function(f){this.$el.fadeOut(c.fxSpeed,f)})}$(c).queue(function(f){this.$el.empty().attr("class",c.className).addClass("state-"+c.model.get("state")).append(d.children());if(this.selectable){this.showSelector(0)}f()});if(e){$(c).queue(function(f){this.$el.fadeIn(c.fxSpeed,f)})}$(c).queue(function(f){this.trigger("rendered",c);if(this.model.inReadyState()){this.trigger("rendered:ready",c)}if(this.draggable){this.draggableOn()}f()});return this},_setUpBehaviors:function(c){c=c||this.$el;make_popup_menus(c);c.find("[title]").tooltip({placement:"bottom"})},_render_titleButtons:function(){return[this._render_displayButton()]},_render_displayButton:function(){if((this.model.get("state")===b.HistoryDatasetAssociation.STATES.NOT_VIEWABLE)||(this.model.get("state")===b.HistoryDatasetAssociation.STATES.DISCARDED)||(this.model.get("state")===b.HistoryDatasetAssociation.STATES.NEW)||(!this.model.get("accessible"))){return null}var d={target:this.linkTarget,classes:"dataset-display"};if(this.model.get("purged")){d.disabled=true;d.title=_l("Cannot display datasets removed from disk")}else{if(this.model.get("state")===b.HistoryDatasetAssociation.STATES.UPLOAD){d.disabled=true;d.title=_l("This dataset must finish uploading before it can be viewed")}else{d.title=_l("View data");d.href=this.urls.display;var c=this;d.onclick=function(){if(Galaxy.frame&&Galaxy.frame.active){Galaxy.frame.add({title:"Data Viewer: "+c.model.get("name"),type:"url",content:c.urls.display})}}}}d.faIcon="fa-eye";return faIconButton(d)},_render_downloadButton:function(){if(this.model.get("purged")||!this.model.hasData()){return null}var d=this.urls,e=this.model.get("meta_files");if(_.isEmpty(e)){return $(['<a href="'+d.download+'" title="'+_l("Download")+'" class="icon-btn">','<span class="fa fa-floppy-o"></span>',"</a>"].join(""))}var f="dataset-"+this.model.get("id")+"-popup",c=['<div popupmenu="'+f+'">','<a href="'+d.download+'">',_l("Download Dataset"),"</a>","<a>"+_l("Additional Files")+"</a>",_.map(e,function(g){return['<a class="action-button" href="',d.meta_download+g.file_type,'">',_l("Download")," ",g.file_type,"</a>"].join("")}).join("\n"),"</div>",'<div class="icon-btn-group">','<a href="'+d.download+'" title="'+_l("Download")+'" class="icon-btn">','<span class="fa fa-floppy-o"></span>','</a><a class="icon-btn popup" id="'+f+'">','<span class="fa fa-caret-down"></span>',"</a>","</div>"].join("\n");return $(c)},_render_showParamsButton:function(){return faIconButton({title:_l("View details"),href:this.urls.show_params,target:this.linkTarget,faIcon:"fa-info-circle"})},_render_body:function(){var d=$('<div>Error: unknown dataset state "'+this.model.get("state")+'".</div>'),c=this["_render_body_"+this.model.get("state")];if(_.isFunction(c)){d=c.call(this)}this._setUpBehaviors(d);if(this.expanded){d.show()}return d},_render_stateBodyHelper:function(c,f){f=f||[];var d=this,e=$(a.templates.body(_.extend(this.model.toJSON(),{body:c})));e.find(".dataset-actions .left").append(_.map(f,function(g){return g.call(d)}));return e},_render_body_new:function(){return this._render_stateBodyHelper("<div>"+_l("This is a new dataset and not all of its data are available yet")+"</div>")},_render_body_noPermission:function(){return this._render_stateBodyHelper("<div>"+_l("You do not have permission to view this dataset")+"</div>")},_render_body_discarded:function(){return this._render_stateBodyHelper("<div>"+_l("The job creating this dataset was cancelled before completion")+"</div>",this.defaultPrimaryActionButtonRenderers)},_render_body_queued:function(){return this._render_stateBodyHelper("<div>"+_l("This job is waiting to run")+"</div>",this.defaultPrimaryActionButtonRenderers)},_render_body_upload:function(){return this._render_stateBodyHelper("<div>"+_l("This dataset is currently uploading")+"</div>")},_render_body_setting_metadata:function(){return this._render_stateBodyHelper("<div>"+_l("Metadata is being auto-detected")+"</div>")},_render_body_running:function(){return this._render_stateBodyHelper("<div>"+_l("This job is currently running")+"</div>",this.defaultPrimaryActionButtonRenderers)},_render_body_paused:function(){return this._render_stateBodyHelper("<div>"+_l('This job is paused. Use the "Resume Paused Jobs" in the history menu to resume')+"</div>",this.defaultPrimaryActionButtonRenderers)},_render_body_error:function(){var c=['<span class="help-text">',_l("An error occurred with this dataset"),":</span>",'<div class="job-error-text">',$.trim(this.model.get("misc_info")),"</div>"].join("");if(!this.model.get("purged")){c="<div>"+this.model.get("misc_blurb")+"</div>"+c}return this._render_stateBodyHelper(c,this.defaultPrimaryActionButtonRenderers.concat([this._render_downloadButton]))},_render_body_empty:function(){return this._render_stateBodyHelper("<div>"+_l("No data")+": <i>"+this.model.get("misc_blurb")+"</i></div>",this.defaultPrimaryActionButtonRenderers)},_render_body_failed_metadata:function(){var c=$('<div class="warningmessagesmall"></div>').append($("<strong/>").text(_l("An error occurred setting the metadata for this dataset"))),d=this._render_body_ok();d.prepend(c);return d},_render_body_ok:function(){var c=this,e=$(a.templates.body(this.model.toJSON())),d=[this._render_downloadButton].concat(this.defaultPrimaryActionButtonRenderers);e.find(".dataset-actions .left").append(_.map(d,function(f){return f.call(c)}));if(this.model.isDeletedOrPurged()){return e}return e},events:{"click .dataset-title-bar":"toggleBodyVisibility","keydown .dataset-title-bar":"toggleBodyVisibility","click .dataset-selector":"toggleSelect",},toggleBodyVisibility:function(f,d){var c=32,e=13;if(f&&(f.type==="keydown")&&!(f.keyCode===c||f.keyCode===e)){return true}var g=this.$el.find(".dataset-body");d=(d===undefined)?(!g.is(":visible")):(d);if(d){this.expandBody()}else{this.collapseBody()}return false},expandBody:function(){var c=this;function d(){c.$el.children(".dataset-body").replaceWith(c._render_body());c.$el.children(".dataset-body").slideDown(c.fxSpeed,function(){c.expanded=true;c.trigger("body-expanded",c.model.get("id"))})}if(this.model.inReadyState()&&!this.model.hasDetails()){this.model.fetch({silent:true}).always(function(e){d()})}else{d()}},collapseBody:function(){var c=this;this.$el.children(".dataset-body").slideUp(c.fxSpeed,function(){c.expanded=false;c.trigger("body-collapsed",c.model.get("id"))})},showSelector:function(e){if(this.$el.find(".dataset-selector").css("display")!=="none"){return}e=(e!==undefined)?(e):(this.fxSpeed);if(this.selected){this.select(null,true)}var d=this,c=32;if(e){this.$el.queue("fx",function(f){$(this).find(".dataset-primary-actions").fadeOut(e,f)});this.$el.queue("fx",function(f){$(this).find(".dataset-selector").show().animate({width:c},e,f);$(this).find(".dataset-title-bar").animate({"margin-left":c},e,f);d.selectable=true;d.trigger("selectable",true,d)})}else{this.$el.find(".dataset-primary-actions").hide();this.$el.find(".dataset-selector").show().css({width:c});this.$el.find(".dataset-title-bar").show().css({"margin-left":c});d.selectable=true;d.trigger("selectable",true,d)}},hideSelector:function(c){c=(c!==undefined)?(c):(this.fxSpeed);this.selectable=false;this.trigger("selectable",false,this);if(c){this.$el.queue("fx",function(d){$(this).find(".dataset-title-bar").show().css({"margin-left":"0"});$(this).find(".dataset-selector").animate({width:"0px"},c,function(){$(this).hide();d()})});this.$el.queue("fx",function(d){$(this).find(".dataset-primary-actions").fadeIn(c,d)})}else{$(this).find(".dataset-selector").css({width:"0px"}).hide();$(this).find(".dataset-primary-actions").show()}},toggleSelector:function(c){if(!this.$el.find(".dataset-selector").is(":visible")){this.showSelector(c)}else{this.hideSelector(c)}},select:function(c){this.$el.find(".dataset-selector span").removeClass("fa-square-o").addClass("fa-check-square-o");if(!this.selected){this.trigger("selected",this);this.selected=true}return false},deselect:function(c){this.$el.find(".dataset-selector span").removeClass("fa-check-square-o").addClass("fa-square-o");if(this.selected){this.trigger("de-selected",this);this.selected=false}return false},toggleSelect:function(c){if(this.selected){this.deselect(c)}else{this.select(c)}},draggableOn:function(){this.draggable=true;this.dragStartHandler=_.bind(this._dragStartHandler,this);this.dragEndHandler=_.bind(this._dragEndHandler,this);var c=this.$el.find(".dataset-title-bar").attr("draggable",true).get(0);c.addEventListener("dragstart",this.dragStartHandler,false);c.addEventListener("dragend",this.dragEndHandler,false)},draggableOff:function(){this.draggable=false;var c=this.$el.find(".dataset-title-bar").attr("draggable",false).get(0);c.removeEventListener("dragstart",this.dragStartHandler,false);c.removeEventListener("dragend",this.dragEndHandler,false)},toggleDraggable:function(){if(this.draggable){this.draggableOff()}else{this.draggableOn()}},_dragStartHandler:function(c){this.trigger("dragstart",this);c.dataTransfer.effectAllowed="move";c.dataTransfer.setData("text",JSON.stringify(this.model.toJSON()));return false},_dragEndHandler:function(c){this.trigger("dragend",this);return false},remove:function(d){var c=this;this.$el.fadeOut(c.fxSpeed,function(){c.$el.remove();c.off();if(d){d()}})},toString:function(){var c=(this.model)?(this.model+""):("(no model)");return"HDABaseView("+c+")"}});a.templates={skeleton:Handlebars.templates["template-hda-skeleton"],body:Handlebars.templates["template-hda-body"]};return{HDABaseView:a}});
\ No newline at end of file
+define(["mvc/dataset/hda-model"],function(b){var a=Backbone.View.extend(LoggableMixin).extend({tagName:"div",className:"dataset hda history-panel-hda",id:function(){return"hda-"+this.model.get("id")},fxSpeed:"fast",initialize:function(c){if(c.logger){this.logger=this.model.logger=c.logger}this.log(this+".initialize:",c);this.defaultPrimaryActionButtonRenderers=[this._render_showParamsButton];this.linkTarget=c.linkTarget||"_blank";this.selectable=c.selectable||false;this.selected=c.selected||false;this.expanded=c.expanded||false;this.draggable=c.draggable||false;this._setUpListeners()},_setUpListeners:function(){this.model.on("change",function(d,c){if(this.model.changedAttributes().state&&this.model.inReadyState()&&this.expanded&&!this.model.hasDetails()){this.model.fetch()}else{this.render()}},this)},render:function(e){e=(e===undefined)?(true):(e);var c=this;this.$el.find("[title]").tooltip("destroy");this.urls=this.model.urls();var d=$(a.templates.skeleton(this.model.toJSON()));d.find(".dataset-primary-actions").append(this._render_titleButtons());d.children(".dataset-body").replaceWith(this._render_body());this._setUpBehaviors(d);if(e){$(c).queue(function(f){this.$el.fadeOut(c.fxSpeed,f)})}$(c).queue(function(f){this.$el.empty().attr("class",c.className).addClass("state-"+c.model.get("state")).append(d.children());if(this.selectable){this.showSelector(0)}f()});if(e){$(c).queue(function(f){this.$el.fadeIn(c.fxSpeed,f)})}$(c).queue(function(f){this.trigger("rendered",c);if(this.model.inReadyState()){this.trigger("rendered:ready",c)}if(this.draggable){this.draggableOn()}f()});return this},_setUpBehaviors:function(c){c=c||this.$el;make_popup_menus(c);c.find("[title]").tooltip({placement:"bottom"})},_render_titleButtons:function(){return[this._render_displayButton()]},_render_displayButton:function(){if((this.model.get("state")===b.HistoryDatasetAssociation.STATES.NOT_VIEWABLE)||(this.model.get("state")===b.HistoryDatasetAssociation.STATES.DISCARDED)||(this.model.get("state")===b.HistoryDatasetAssociation.STATES.NEW)||(!this.model.get("accessible"))){return null}var d={target:this.linkTarget,classes:"dataset-display"};if(this.model.get("purged")){d.disabled=true;d.title=_l("Cannot display datasets removed from disk")}else{if(this.model.get("state")===b.HistoryDatasetAssociation.STATES.UPLOAD){d.disabled=true;d.title=_l("This dataset must finish uploading before it can be viewed")}else{d.title=_l("View data");d.href=this.urls.display;var c=this;d.onclick=function(){if(Galaxy.frame&&Galaxy.frame.active){Galaxy.frame.add({title:"Data Viewer: "+c.model.get("name"),type:"url",content:c.urls.display})}}}}d.faIcon="fa-eye";return faIconButton(d)},_render_downloadButton:function(){if(this.model.get("purged")||!this.model.hasData()){return null}var d=this.urls,e=this.model.get("meta_files");if(_.isEmpty(e)){return $(['<a href="'+d.download+'" title="'+_l("Download")+'" class="icon-btn">','<span class="fa fa-floppy-o"></span>',"</a>"].join(""))}var f="dataset-"+this.model.get("id")+"-popup",c=['<div popupmenu="'+f+'">','<a href="'+d.download+'">',_l("Download Dataset"),"</a>","<a>"+_l("Additional Files")+"</a>",_.map(e,function(g){return['<a class="action-button" href="',d.meta_download+g.file_type,'">',_l("Download")," ",g.file_type,"</a>"].join("")}).join("\n"),"</div>",'<div class="icon-btn-group">','<a href="'+d.download+'" title="'+_l("Download")+'" class="icon-btn">','<span class="fa fa-floppy-o"></span>','</a><a class="icon-btn popup" id="'+f+'">','<span class="fa fa-caret-down"></span>',"</a>","</div>"].join("\n");return $(c)},_render_showParamsButton:function(){return faIconButton({title:_l("View details"),href:this.urls.show_params,target:this.linkTarget,faIcon:"fa-info-circle"})},_render_body:function(){var d=$('<div>Error: unknown dataset state "'+this.model.get("state")+'".</div>'),c=this["_render_body_"+this.model.get("state")];if(_.isFunction(c)){d=c.call(this)}this._setUpBehaviors(d);if(this.expanded){d.show()}return d},_render_stateBodyHelper:function(c,f){f=f||[];var d=this,e=$(a.templates.body(_.extend(this.model.toJSON(),{body:c})));e.find(".dataset-actions .left").append(_.map(f,function(g){return g.call(d)}));return e},_render_body_new:function(){return this._render_stateBodyHelper("<div>"+_l("This is a new dataset and not all of its data are available yet")+"</div>")},_render_body_noPermission:function(){return this._render_stateBodyHelper("<div>"+_l("You do not have permission to view this dataset")+"</div>")},_render_body_discarded:function(){return this._render_stateBodyHelper("<div>"+_l("The job creating this dataset was cancelled before completion")+"</div>",this.defaultPrimaryActionButtonRenderers)},_render_body_queued:function(){return this._render_stateBodyHelper("<div>"+_l("This job is waiting to run")+"</div>",this.defaultPrimaryActionButtonRenderers)},_render_body_upload:function(){return this._render_stateBodyHelper("<div>"+_l("This dataset is currently uploading")+"</div>")},_render_body_setting_metadata:function(){return this._render_stateBodyHelper("<div>"+_l("Metadata is being auto-detected")+"</div>")},_render_body_running:function(){return this._render_stateBodyHelper("<div>"+_l("This job is currently running")+"</div>",this.defaultPrimaryActionButtonRenderers)},_render_body_paused:function(){return this._render_stateBodyHelper("<div>"+_l('This job is paused. Use the "Resume Paused Jobs" in the history menu to resume')+"</div>",this.defaultPrimaryActionButtonRenderers)},_render_body_error:function(){var c=['<span class="help-text">',_l("An error occurred with this dataset"),":</span>",'<div class="job-error-text">',$.trim(this.model.get("misc_info")),"</div>"].join("");if(!this.model.get("purged")){c="<div>"+this.model.get("misc_blurb")+"</div>"+c}return this._render_stateBodyHelper(c,[this._render_downloadButton].concat(this.defaultPrimaryActionButtonRenderers))},_render_body_empty:function(){return this._render_stateBodyHelper("<div>"+_l("No data")+": <i>"+this.model.get("misc_blurb")+"</i></div>",this.defaultPrimaryActionButtonRenderers)},_render_body_failed_metadata:function(){var c=$('<div class="warningmessagesmall"></div>').append($("<strong/>").text(_l("An error occurred setting the metadata for this dataset"))),d=this._render_body_ok();d.prepend(c);return d},_render_body_ok:function(){var c=this,e=$(a.templates.body(this.model.toJSON())),d=[this._render_downloadButton].concat(this.defaultPrimaryActionButtonRenderers);e.find(".dataset-actions .left").append(_.map(d,function(f){return f.call(c)}));if(this.model.isDeletedOrPurged()){return e}return e},events:{"click .dataset-title-bar":"toggleBodyVisibility","keydown .dataset-title-bar":"toggleBodyVisibility","click .dataset-selector":"toggleSelect",},toggleBodyVisibility:function(f,d){var c=32,e=13;if(f&&(f.type==="keydown")&&!(f.keyCode===c||f.keyCode===e)){return true}var g=this.$el.find(".dataset-body");d=(d===undefined)?(!g.is(":visible")):(d);if(d){this.expandBody()}else{this.collapseBody()}return false},expandBody:function(){var c=this;function d(){c.$el.children(".dataset-body").replaceWith(c._render_body());c.$el.children(".dataset-body").slideDown(c.fxSpeed,function(){c.expanded=true;c.trigger("body-expanded",c.model.get("id"))})}if(this.model.inReadyState()&&!this.model.hasDetails()){this.model.fetch({silent:true}).always(function(e){d()})}else{d()}},collapseBody:function(){var c=this;this.$el.children(".dataset-body").slideUp(c.fxSpeed,function(){c.expanded=false;c.trigger("body-collapsed",c.model.get("id"))})},showSelector:function(e){if(this.$el.find(".dataset-selector").css("display")!=="none"){return}e=(e!==undefined)?(e):(this.fxSpeed);if(this.selected){this.select(null,true)}var d=this,c=32;if(e){this.$el.queue("fx",function(f){$(this).find(".dataset-primary-actions").fadeOut(e,f)});this.$el.queue("fx",function(f){$(this).find(".dataset-selector").show().animate({width:c},e,f);$(this).find(".dataset-title-bar").animate({"margin-left":c},e,f);d.selectable=true;d.trigger("selectable",true,d)})}else{this.$el.find(".dataset-primary-actions").hide();this.$el.find(".dataset-selector").show().css({width:c});this.$el.find(".dataset-title-bar").show().css({"margin-left":c});d.selectable=true;d.trigger("selectable",true,d)}},hideSelector:function(c){c=(c!==undefined)?(c):(this.fxSpeed);this.selectable=false;this.trigger("selectable",false,this);if(c){this.$el.queue("fx",function(d){$(this).find(".dataset-title-bar").show().css({"margin-left":"0"});$(this).find(".dataset-selector").animate({width:"0px"},c,function(){$(this).hide();d()})});this.$el.queue("fx",function(d){$(this).find(".dataset-primary-actions").fadeIn(c,d)})}else{$(this).find(".dataset-selector").css({width:"0px"}).hide();$(this).find(".dataset-primary-actions").show()}},toggleSelector:function(c){if(!this.$el.find(".dataset-selector").is(":visible")){this.showSelector(c)}else{this.hideSelector(c)}},select:function(c){this.$el.find(".dataset-selector span").removeClass("fa-square-o").addClass("fa-check-square-o");if(!this.selected){this.trigger("selected",this);this.selected=true}return false},deselect:function(c){this.$el.find(".dataset-selector span").removeClass("fa-check-square-o").addClass("fa-square-o");if(this.selected){this.trigger("de-selected",this);this.selected=false}return false},toggleSelect:function(c){if(this.selected){this.deselect(c)}else{this.select(c)}},draggableOn:function(){this.draggable=true;this.dragStartHandler=_.bind(this._dragStartHandler,this);this.dragEndHandler=_.bind(this._dragEndHandler,this);var c=this.$el.find(".dataset-title-bar").attr("draggable",true).get(0);c.addEventListener("dragstart",this.dragStartHandler,false);c.addEventListener("dragend",this.dragEndHandler,false)},draggableOff:function(){this.draggable=false;var c=this.$el.find(".dataset-title-bar").attr("draggable",false).get(0);c.removeEventListener("dragstart",this.dragStartHandler,false);c.removeEventListener("dragend",this.dragEndHandler,false)},toggleDraggable:function(){if(this.draggable){this.draggableOff()}else{this.draggableOn()}},_dragStartHandler:function(c){this.trigger("dragstart",this);c.dataTransfer.effectAllowed="move";c.dataTransfer.setData("text",JSON.stringify(this.model.toJSON()));return false},_dragEndHandler:function(c){this.trigger("dragend",this);return false},remove:function(d){var c=this;this.$el.fadeOut(c.fxSpeed,function(){c.$el.remove();c.off();if(d){d()}})},toString:function(){var c=(this.model)?(this.model+""):("(no model)");return"HDABaseView("+c+")"}});a.templates={skeleton:Handlebars.templates["template-hda-skeleton"],body:Handlebars.templates["template-hda-body"]};return{HDABaseView:a}});
\ No newline at end of file
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0
                            
                          
                          
                            
    
                          
                        
                    
                        
                            
                                
                            
                            commit/galaxy-central: carlfeberhard: History panel: do not update title when empty string or unchanged
                        
                        
by commits-noreply@bitbucket.org 19 Dec '13
                    by commits-noreply@bitbucket.org 19 Dec '13
19 Dec '13
                    
                        1 new commit in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/4fc04317cef3/
Changeset:   4fc04317cef3
User:        carlfeberhard
Date:        2013-12-19 16:39:55
Summary:     History panel: do not update title when empty string or unchanged
Affected #:  2 files
diff -r 67fe4b091f0ad66d68d06a36b8fb645aa445d032 -r 4fc04317cef36f002e071d391523450a7a956232 static/scripts/mvc/history/history-panel.js
--- a/static/scripts/mvc/history/history-panel.js
+++ b/static/scripts/mvc/history/history-panel.js
@@ -698,11 +698,16 @@
             .attr( 'title', _l( 'Click to rename history' ) ).tooltip({ placement: 'bottom' })
             .make_text_editable({
                 on_finish: function( newName ){
-                    $where.find( '.history-name' ).text( newName );
-                    panel.model.save({ name: newName })
-                        .fail( function(){
-                            $where.find( '.history-name' ).text( panel.model.previous( 'name' ) );
-                        });
+                    var previousName = panel.model.get( 'name' );
+                    if( newName && newName !== previousName ){
+                        panel.$el.find( '.history-name' ).text( newName );
+                        panel.model.save({ name: newName })
+                            .fail( function(){
+                                panel.$el.find( '.history-name' ).text( panel.model.previous( 'name' ) );
+                            });
+                    } else {
+                        panel.$el.find( '.history-name' ).text( previousName );
+                    }
                 }
             });
         this._setUpDatasetActionsPopup( $where );
diff -r 67fe4b091f0ad66d68d06a36b8fb645aa445d032 -r 4fc04317cef36f002e071d391523450a7a956232 static/scripts/packed/mvc/history/history-panel.js
--- a/static/scripts/packed/mvc/history/history-panel.js
+++ b/static/scripts/packed/mvc/history/history-panel.js
@@ -1,1 +1,1 @@
-define(["mvc/history/history-model","mvc/dataset/hda-model","mvc/dataset/hda-base","mvc/dataset/hda-edit"],function(c,f,a,i){var g=SessionStorageModel.extend({defaults:{searching:false,tagsEditorShown:false,annotationEditorShown:false},toString:function(){return"HistoryPanelPrefs("+JSON.stringify(this.toJSON())+")"}});g.storageKey=function h(){return("history-panel")};var d=SessionStorageModel.extend({defaults:{expandedHdas:{},show_deleted:false,show_hidden:false},addExpandedHda:function(j){this.save("expandedHdas",_.extend(this.get("expandedHdas"),_.object([j],[true])))},removeExpandedHda:function(j){this.save("expandedHdas",_.omit(this.get("expandedHdas"),j))},toString:function(){return"HistoryPrefs("+this.id+")"}});d.historyStorageKey=function b(j){if(!j){throw new Error("HistoryPrefs.historyStorageKey needs valid id: "+j)}return("history:"+j)};var e=Backbone.View.extend(LoggableMixin).extend({defaultHDAViewClass:i.HDAEditView,tagName:"div",className:"history-panel",fxSpeed:"fast",datasetsSelector:".datasets-list",emptyMsgSelector:".empty-history-message",msgsSelector:".message-container",initialize:function(j){j=j||{};if(j.logger){this.logger=j.logger}this.log(this+".initialize:",j);this.HDAViewClass=j.HDAViewClass||this.defaultHDAViewClass;this.linkTarget=j.linkTarget||"_blank";this.hdaViews={};this.indicator=new LoadingIndicator(this.$el);this.preferences=new g(_.extend({id:g.storageKey()},_.pick(j,_.keys(g.prototype.defaults))));this.filters=[];this.selecting=j.selecting||false;this.annotationEditorShown=j.annotationEditorShown||false;this._setUpListeners();if(this.model){this._setUpWebStorage(j.initiallyExpanded,j.show_deleted,j.show_hidden);this._setUpModelEventHandlers()}if(j.onready){j.onready.call(this)}},_setUpListeners:function(){this.on("error",function(k,n,j,m,l){this.errorHandler(k,n,j,m,l)});this.on("loading-history",function(){this.showLoadingIndicator("loading history...")});this.on("loading-done",function(){this.hideLoadingIndicator()});this.once("rendered",function(){this.trigger("rendered:initial",this);return false});this.on("switched-history current-history new-history",function(){if(_.isEmpty(this.hdaViews)){this.trigger("empty-history",this)}});if(this.logger){this.on("all",function(j){this.log(this+"",arguments)},this)}},errorHandler:function(l,o,k,n,m){var j=this._parseErrorMessage(l,o,k,n,m);if(o&&o.status===0&&o.readyState===0){}else{if(o&&o.status===502){}else{if(!this.$el.find(this.msgsSelector).is(":visible")){this.once("rendered",function(){this.displayMessage("error",j.message,j.details)})}else{this.displayMessage("error",j.message,j.details)}}}},_parseErrorMessage:function(m,q,l,p,o){var k=Galaxy.currUser,j={message:this._bePolite(p),details:{user:(k instanceof User)?(k.toJSON()):(k+""),source:(m instanceof Backbone.Model)?(m.toJSON()):(m+""),xhr:q,options:(q)?(_.omit(l,"xhr")):(l)}};_.extend(j.details,o||{});if(q&&_.isFunction(q.getAllResponseHeaders)){var n=q.getAllResponseHeaders();n=_.compact(n.split("\n"));n=_.map(n,function(r){return r.split(": ")});j.details.xhr.responseHeaders=_.object(n)}return j},_bePolite:function(j){j=j||_l("An error occurred while getting updates from the server");return j+". "+_l("Please contact a Galaxy administrator if the problem persists.")},loadCurrentHistory:function(k){var j=this;return this.loadHistoryWithHDADetails("current",k).then(function(m,l){j.trigger("current-history",j)})},switchToHistory:function(m,l){var j=this,k=function(){return jQuery.post(galaxy_config.root+"api/histories/"+m+"/set_as_current")};return this.loadHistoryWithHDADetails(m,l,k).then(function(o,n){j.trigger("switched-history",j)})},createNewHistory:function(l){if(!Galaxy||!Galaxy.currUser||Galaxy.currUser.isAnonymous()){this.displayMessage("error",_l("You must be logged in to create histories"));return $.when()}var j=this,k=function(){return jQuery.post(galaxy_config.root+"api/histories",{current:true})};return this.loadHistory(undefined,l,k).then(function(n,m){j.trigger("new-history",j)})},loadHistoryWithHDADetails:function(m,l,k,o){var j=this,n=function(p){return j.getExpandedHdaIds(p.id)};return this.loadHistory(m,l,k,o,n)},loadHistory:function(m,l,k,p,n){this.trigger("loading-history",this);l=l||{};var j=this;var o=c.History.getHistoryData(m,{historyFn:k,hdaFn:p,hdaDetailIds:l.initiallyExpanded||n});return this._loadHistoryFromXHR(o,l).fail(function(s,q,r){j.trigger("error",j,s,l,_l("An error was encountered while "+q),{historyId:m,history:r||{}})}).always(function(){j.trigger("loading-done",j)})},_loadHistoryFromXHR:function(l,k){var j=this;l.then(function(m,n){j.setModel(m,n,k)});l.fail(function(n,m){j.render()});return l},setModel:function(l,j,k){k=k||{};if(this.model){this.model.clearUpdateTimeout();this.stopListening(this.model);this.stopListening(this.model.hdas)}this.hdaViews={};if(Galaxy&&Galaxy.currUser){l.user=Galaxy.currUser.toJSON()}this.model=new c.History(l,j,k);this._setUpWebStorage(k.initiallyExpanded,k.show_deleted,k.show_hidden);this._setUpModelEventHandlers();this.trigger("new-model",this);this.render();return this},_setUpWebStorage:function(k,j,l){this.storage=new d({id:d.historyStorageKey(this.model.get("id"))});if(_.isObject(k)){this.storage.set("exandedHdas",k)}if(_.isBoolean(j)){this.storage.set("show_deleted",j)}if(_.isBoolean(l)){this.storage.set("show_hidden",l)}this.trigger("new-storage",this.storage,this);this.log(this+" (init'd) storage:",this.storage.get())},clearWebStorage:function(){for(var j in sessionStorage){if(j.indexOf("history:")===0){sessionStorage.removeItem(j)}}},getStoredOptions:function(k){if(!k||k==="current"){return(this.storage)?(this.storage.get()):({})}var j=sessionStorage.getItem(d.historyStorageKey(k));return(j===null)?({}):(JSON.parse(j))},getExpandedHdaIds:function(j){var k=this.getStoredOptions(j).expandedHdas;return((_.isEmpty(k))?([]):(_.keys(k)))},_setUpModelEventHandlers:function(){this.model.on("error error:hdas",function(k,m,j,l){this.errorHandler(k,m,j,l)},this);this.model.on("change:nice_size",this.updateHistoryDiskSize,this);if(Galaxy&&Galaxy.quotaMeter){this.listenTo(this.model,"change:nice_size",function(){Galaxy.quotaMeter.update()})}this.model.hdas.on("add",this.addHdaView,this);this.model.hdas.on("change:deleted",this.handleHdaDeletionChange,this);this.model.hdas.on("change:visible",this.handleHdaVisibleChange,this);this.model.hdas.on("change:purged",function(j){this.model.fetch()},this);this.model.hdas.on("state:ready",function(k,l,j){if((!k.get("visible"))&&(!this.storage.get("show_hidden"))){this.removeHdaView(this.hdaViews[k.id])}},this)},render:function(l,m){l=(l===undefined)?(this.fxSpeed):(l);var j=this,k;if(this.model){k=this.renderModel()}else{k=this.renderWithoutModel()}$(j).queue("fx",[function(n){if(l&&j.$el.is(":visible")){j.$el.fadeOut(l,n)}else{n()}},function(n){j.$el.empty();if(k){j.$el.append(k.children());j.renderBasedOnPrefs()}n()},function(n){if(l&&!j.$el.is(":visible")){j.$el.fadeIn(l,n)}else{n()}},function(n){if(m){m.call(this)}j.trigger("rendered",this);n()}]);return this},renderWithoutModel:function(){var j=$("<div/>"),k=$("<div/>").addClass("message-container").css({"margin-left":"4px","margin-right":"4px"});return j.append(k)},renderModel:function(){var j=$("<div/>");if(!Galaxy||!Galaxy.currUser||Galaxy.currUser.isAnonymous()){j.append(e.templates.anonHistoryPanel(this.model.toJSON()))}else{j.append(e.templates.historyPanel(this.model.toJSON()));if(Galaxy.currUser.id&&Galaxy.currUser.id===this.model.get("user_id")){this._renderTags(j);this._renderAnnotation(j)}}j.find(".history-secondary-actions").prepend(this._renderSearchButton());this._setUpBehaviours(j);this.renderHdas(j);return j},renderBasedOnPrefs:function(){if(this.preferences.get("searching")){this.showSearchControls(0)}},_renderTags:function(j){var k=this;this.tagsEditor=new TagsEditor({model:this.model,el:j.find(".history-controls .tags-display"),onshowFirstTime:function(){this.render()},onshow:function(){k.preferences.set("tagsEditorShown",true);k.toggleHDATagEditors(true,k.fxSpeed)},onhide:function(){k.preferences.set("tagsEditorShown",false);k.toggleHDATagEditors(false,k.fxSpeed)},$activator:faIconButton({title:_l("Edit history tags"),classes:"history-tag-btn",faIcon:"fa-tags"}).appendTo(j.find(".history-secondary-actions"))});if(this.preferences.get("tagsEditorShown")){this.tagsEditor.toggle(true)}},_renderAnnotation:function(j){var k=this;this.annotationEditor=new AnnotationEditor({model:this.model,el:j.find(".history-controls .annotation-display"),onshowFirstTime:function(){this.render()},onshow:function(){k.preferences.set("annotationEditorShown",true);k.toggleHDAAnnotationEditors(true,k.fxSpeed)},onhide:function(){k.preferences.set("annotationEditorShown",false);k.toggleHDAAnnotationEditors(false,k.fxSpeed)},$activator:faIconButton({title:_l("Edit history Annotation"),classes:"history-annotate-btn",faIcon:"fa-comment"}).appendTo(j.find(".history-secondary-actions"))});if(this.preferences.get("annotationEditorShown")){this.annotationEditor.toggle(true)}},_renderSearchButton:function(j){return faIconButton({title:_l("Search datasets"),classes:"history-search-btn",faIcon:"fa-search"})},_renderSelectButton:function(j){return faIconButton({title:_l("Operations on multiple datasets"),classes:"history-select-btn",faIcon:"fa-check-square-o"})},_setUpBehaviours:function(j){j=j||this.$el;j.find("[title]").tooltip({placement:"bottom"});if((!this.model)||(!Galaxy.currUser||Galaxy.currUser.isAnonymous())||(Galaxy.currUser.id!==this.model.get("user_id"))){return}var k=this;j.find(".history-name").attr("title",_l("Click to rename history")).tooltip({placement:"bottom"}).make_text_editable({on_finish:function(l){j.find(".history-name").text(l);k.model.save({name:l}).fail(function(){j.find(".history-name").text(k.model.previous("name"))})}});this._setUpDatasetActionsPopup(j)},_setUpDatasetActionsPopup:function(j){var k=this;(new PopupMenu(j.find(".history-dataset-action-popup-btn"),[{html:_l("Hide datasets"),func:function(){var l=f.HistoryDatasetAssociation.prototype.hide;k.getSelectedHdaCollection().ajaxQueue(l)}},{html:_l("Unhide datasets"),func:function(){var l=f.HistoryDatasetAssociation.prototype.unhide;k.getSelectedHdaCollection().ajaxQueue(l)}},{html:_l("Delete datasets"),func:function(){var l=f.HistoryDatasetAssociation.prototype["delete"];k.getSelectedHdaCollection().ajaxQueue(l)}},{html:_l("Undelete datasets"),func:function(){var l=f.HistoryDatasetAssociation.prototype.undelete;k.getSelectedHdaCollection().ajaxQueue(l)}},{html:_l("Permanently delete datasets"),func:function(){if(confirm(_l("This will permanently remove the data in your datasets. Are you sure?"))){var l=f.HistoryDatasetAssociation.prototype.purge;k.getSelectedHdaCollection().ajaxQueue(l)}}}]))},refreshHdas:function(k,j){if(this.model){return this.model.refresh(k,j)}return $.when()},addHdaView:function(m){this.log("add."+this,m);var k=this;if(!m.isVisible(this.storage.get("show_deleted"),this.storage.get("show_hidden"))){return}$({}).queue([function l(o){var n=k.$el.find(k.emptyMsgSelector);if(n.is(":visible")){n.fadeOut(k.fxSpeed,o)}else{o()}},function j(o){k.scrollToTop();var n=k.$el.find(k.datasetsSelector);k.createHdaView(m).$el.hide().prependTo(n).slideDown(k.fxSpeed)}])},createHdaView:function(l){var k=l.get("id"),j=this.storage.get("expandedHdas")[k],m=new this.HDAViewClass({model:l,linkTarget:this.linkTarget,expanded:j,tagsEditorShown:this.preferences.get("tagsEditorShown"),annotationEditorShown:this.preferences.get("annotationEditorShown"),selectable:this.selecting,hasUser:this.model.ownedByCurrUser(),logger:this.logger});this._setUpHdaListeners(m);this.hdaViews[k]=m;return m.render()},_setUpHdaListeners:function(k){var j=this;k.on("body-expanded",function(l){j.storage.addExpandedHda(l)});k.on("body-collapsed",function(l){j.storage.removeExpandedHda(l)});k.on("error",function(m,o,l,n){j.errorHandler(m,o,l,n)})},handleHdaDeletionChange:function(j){if(j.get("deleted")&&!this.storage.get("show_deleted")){this.removeHdaView(this.hdaViews[j.id])}},handleHdaVisibleChange:function(j){if(j.hidden()&&!this.storage.get("show_hidden")){this.removeHdaView(this.hdaViews[j.id])}},removeHdaView:function(k){if(!k){return}var j=this;k.$el.fadeOut(j.fxSpeed,function(){k.off();k.remove();delete j.hdaViews[k.model.id];if(_.isEmpty(j.hdaViews)){j.$el.find(j.emptyMsgSelector).fadeIn(j.fxSpeed,function(){j.trigger("empty-history",j)})}})},renderHdas:function(l){l=l||this.$el;this.hdaViews={};var k=this,j=l.find(this.datasetsSelector),m=this.model.hdas.getVisible(this.storage.get("show_deleted"),this.storage.get("show_hidden"),this.filters);j.empty();if(m.length){m.each(function(n){j.prepend(k.createHdaView(n).$el)});l.find(this.emptyMsgSelector).hide()}else{l.find(this.emptyMsgSelector).show()}return this.hdaViews},toggleHDATagEditors:function(j){var k=arguments;_.each(this.hdaViews,function(l){if(l.tagsEditor){l.tagsEditor.toggle.apply(l.tagsEditor,k)}})},toggleHDAAnnotationEditors:function(j){var k=arguments;_.each(this.hdaViews,function(l){if(l.annotationEditor){l.annotationEditor.toggle.apply(l.annotationEditor,k)}})},events:{"click .message-container":"clearMessages","click .history-search-btn":"toggleSearchControls","click .history-select-btn":function(j){this.toggleSelectors(this.fxSpeed)},"click .history-select-all-datasets-btn":"selectAllDatasets"},updateHistoryDiskSize:function(){this.$el.find(".history-size").text(this.model.get("nice_size"))},collapseAllHdaBodies:function(){_.each(this.hdaViews,function(j){j.toggleBodyVisibility(null,false)});this.storage.set("expandedHdas",{})},toggleShowDeleted:function(){this.storage.set("show_deleted",!this.storage.get("show_deleted"));this.renderHdas();return this.storage.get("show_deleted")},toggleShowHidden:function(){this.storage.set("show_hidden",!this.storage.get("show_hidden"));this.renderHdas();return this.storage.get("show_hidden")},renderSearchControls:function(k){var l=this;function n(o){l.searchFor=o;l.filters=[function(p){return p.matchesAll(l.searchFor)}];l.trigger("search:searching",o,l);l.renderHdas()}function j(o){if(l.model.hdas.haveDetails()){n(o);return}l.$el.find(".history-search-controls").searchInput("toggle-loading");l.model.hdas.fetchAllDetails({silent:true}).always(function(){l.$el.find(".history-search-controls").searchInput("toggle-loading")}).done(function(){n(o)})}function m(){l.searchFor="";l.filters=[];l.trigger("search:clear",l);l.renderHdas()}return k.searchInput({initialVal:l.searchFor,name:"history-search",placeholder:"search datasets",classes:"history-search",onfirstsearch:j,onsearch:n,onclear:m})},showSearchControls:function(l){l=(l===undefined)?(this.fxSpeed):(l);var j=this,k=this.$el.find(".history-search-controls");if(!k.children().size()){k=this.renderSearchControls(k).hide()}k.show(l,function(){$(this).find("input").focus();j.preferences.set("searching",true)})},hideSearchControls:function(){speed=(speed===undefined)?(this.fxSpeed):(speed);var j=this;this.$el.find(".history-search-controls").hide(speed,function(){j.preferences.set("searching",false)})},toggleSearchControls:function(j){speed=(jQuery.type(j)==="number")?(j):(this.fxSpeed);if(this.$el.find(".history-search-controls").is(":visible")){this.hideSearchControls(speed)}else{this.showSearchControls(speed)}},showSelectors:function(j){this.selecting=true;this.$el.find(".history-dataset-actions").slideDown(j);_.each(this.hdaViews,function(k){k.showSelector(j)})},hideSelectors:function(j){this.selecting=false;this.$el.find(".history-dataset-actions").slideUp(j);_.each(this.hdaViews,function(k){k.hideSelector(j)})},toggleSelectors:function(j){if(!this.selecting){this.showSelectors(j)}else{this.hideSelectors(j)}},selectAllDatasets:function(k){var j=this.$el.find(".history-select-all-datasets-btn");currMode=j.data("mode");if(currMode==="select"){_.each(this.hdaViews,function(l){l.select(k)});j.data("mode","deselect");j.text(_l("De-select all"))}else{if(currMode==="deselect"){_.each(this.hdaViews,function(l){l.deselect(k)});j.data("mode","select");j.text(_l("Select all"))}}},getSelectedHdaViews:function(){return _.filter(this.hdaViews,function(j){return j.selected})},getSelectedHdaCollection:function(){return new f.HDACollection(_.map(this.getSelectedHdaViews(),function(j){return j.model}),{historyId:this.model.id})},showLoadingIndicator:function(k,j,l){j=(j!==undefined)?(j):(this.fxSpeed);if(!this.indicator){this.indicator=new LoadingIndicator(this.$el,this.$el.parent())}if(!this.$el.is(":visible")){this.indicator.show(0,l)}else{this.$el.fadeOut(j);this.indicator.show(k,j,l)}},hideLoadingIndicator:function(j,k){j=(j!==undefined)?(j):(this.fxSpeed);if(this.indicator){this.indicator.hide(j,k)}},displayMessage:function(o,p,n){var l=this;this.scrollToTop();var m=this.$el.find(this.msgsSelector),j=$("<div/>").addClass(o+"message").html(p);if(!_.isEmpty(n)){var k=$('<a href="javascript:void(0)">Details</a>').click(function(){Galaxy.modal.show(l.messageToModalOptions(o,p,n));return false});j.append(" ",k)}return m.html(j)},messageToModalOptions:function(n,p,m){var j=this,o=$("<div/>"),l={title:"Details"};function k(q){q=_.omit(q,_.functions(q));return["<table>",_.map(q,function(s,r){s=(_.isObject(s))?(k(s)):(s);return'<tr><td style="vertical-align: top; color: grey">'+r+'</td><td style="padding-left: 8px">'+s+"</td></tr>"}).join(""),"</table>"].join("")}if(_.isObject(m)){l.body=o.append(k(m))}else{l.body=o.html(m)}l.buttons={Ok:function(){Galaxy.modal.hide();j.clearMessages()}};return l},clearMessages:function(){var j=this.$el.find(this.msgsSelector);j.empty()},scrollPosition:function(){return this.$el.parent().scrollTop()},scrollTo:function(j){this.$el.parent().scrollTop(j)},scrollToTop:function(){this.$el.parent().scrollTop(0);return this},scrollIntoView:function(k,l){if(!l){this.$el.parent().parent().scrollTop(k);return this}var j=window,m=this.$el.parent().parent(),o=$(j).innerHeight(),n=(o/2)-(l/2);$(m).scrollTop(k-n);return this},scrollToId:function(k){if((!k)||(!this.hdaViews[k])){return this}var j=this.hdaViews[k].$el;this.scrollIntoView(j.offset().top,j.outerHeight());return this},scrollToHid:function(j){var k=this.model.hdas.getByHid(j);if(!k){return this}return this.scrollToId(k.id)},connectToQuotaMeter:function(j){if(!j){return this}this.listenTo(j,"quota:over",this.showQuotaMessage);this.listenTo(j,"quota:under",this.hideQuotaMessage);this.on("rendered rendered:initial",function(){if(j&&j.isOverQuota()){this.showQuotaMessage()}});return this},showQuotaMessage:function(){var j=this.$el.find(".quota-message");if(j.is(":hidden")){j.slideDown(this.fxSpeed)}},hideQuotaMessage:function(){var j=this.$el.find(".quota-message");if(!j.is(":hidden")){j.slideUp(this.fxSpeed)}},connectToOptionsMenu:function(j){if(!j){return this}this.on("new-storage",function(l,k){if(j&&l){j.findItemByHtml(_l("Include Deleted Datasets")).checked=l.get("show_deleted");j.findItemByHtml(_l("Include Hidden Datasets")).checked=l.get("show_hidden")}});return this},toString:function(){return"HistoryPanel("+((this.model)?(this.model.get("name")):(""))+")"}});e.templates={historyPanel:Handlebars.templates["template-history-historyPanel"],anonHistoryPanel:Handlebars.templates["template-history-historyPanel-anon"]};return{HistoryPanel:e}});
\ No newline at end of file
+define(["mvc/history/history-model","mvc/dataset/hda-model","mvc/dataset/hda-base","mvc/dataset/hda-edit"],function(c,f,a,i){var g=SessionStorageModel.extend({defaults:{searching:false,tagsEditorShown:false,annotationEditorShown:false},toString:function(){return"HistoryPanelPrefs("+JSON.stringify(this.toJSON())+")"}});g.storageKey=function h(){return("history-panel")};var d=SessionStorageModel.extend({defaults:{expandedHdas:{},show_deleted:false,show_hidden:false},addExpandedHda:function(j){this.save("expandedHdas",_.extend(this.get("expandedHdas"),_.object([j],[true])))},removeExpandedHda:function(j){this.save("expandedHdas",_.omit(this.get("expandedHdas"),j))},toString:function(){return"HistoryPrefs("+this.id+")"}});d.historyStorageKey=function b(j){if(!j){throw new Error("HistoryPrefs.historyStorageKey needs valid id: "+j)}return("history:"+j)};var e=Backbone.View.extend(LoggableMixin).extend({defaultHDAViewClass:i.HDAEditView,tagName:"div",className:"history-panel",fxSpeed:"fast",datasetsSelector:".datasets-list",emptyMsgSelector:".empty-history-message",msgsSelector:".message-container",initialize:function(j){j=j||{};if(j.logger){this.logger=j.logger}this.log(this+".initialize:",j);this.HDAViewClass=j.HDAViewClass||this.defaultHDAViewClass;this.linkTarget=j.linkTarget||"_blank";this.hdaViews={};this.indicator=new LoadingIndicator(this.$el);this.preferences=new g(_.extend({id:g.storageKey()},_.pick(j,_.keys(g.prototype.defaults))));this.filters=[];this.selecting=j.selecting||false;this.annotationEditorShown=j.annotationEditorShown||false;this._setUpListeners();if(this.model){this._setUpWebStorage(j.initiallyExpanded,j.show_deleted,j.show_hidden);this._setUpModelEventHandlers()}if(j.onready){j.onready.call(this)}},_setUpListeners:function(){this.on("error",function(k,n,j,m,l){this.errorHandler(k,n,j,m,l)});this.on("loading-history",function(){this.showLoadingIndicator("loading history...")});this.on("loading-done",function(){this.hideLoadingIndicator()});this.once("rendered",function(){this.trigger("rendered:initial",this);return false});this.on("switched-history current-history new-history",function(){if(_.isEmpty(this.hdaViews)){this.trigger("empty-history",this)}});if(this.logger){this.on("all",function(j){this.log(this+"",arguments)},this)}},errorHandler:function(l,o,k,n,m){var j=this._parseErrorMessage(l,o,k,n,m);if(o&&o.status===0&&o.readyState===0){}else{if(o&&o.status===502){}else{if(!this.$el.find(this.msgsSelector).is(":visible")){this.once("rendered",function(){this.displayMessage("error",j.message,j.details)})}else{this.displayMessage("error",j.message,j.details)}}}},_parseErrorMessage:function(m,q,l,p,o){var k=Galaxy.currUser,j={message:this._bePolite(p),details:{user:(k instanceof User)?(k.toJSON()):(k+""),source:(m instanceof Backbone.Model)?(m.toJSON()):(m+""),xhr:q,options:(q)?(_.omit(l,"xhr")):(l)}};_.extend(j.details,o||{});if(q&&_.isFunction(q.getAllResponseHeaders)){var n=q.getAllResponseHeaders();n=_.compact(n.split("\n"));n=_.map(n,function(r){return r.split(": ")});j.details.xhr.responseHeaders=_.object(n)}return j},_bePolite:function(j){j=j||_l("An error occurred while getting updates from the server");return j+". "+_l("Please contact a Galaxy administrator if the problem persists.")},loadCurrentHistory:function(k){var j=this;return this.loadHistoryWithHDADetails("current",k).then(function(m,l){j.trigger("current-history",j)})},switchToHistory:function(m,l){var j=this,k=function(){return jQuery.post(galaxy_config.root+"api/histories/"+m+"/set_as_current")};return this.loadHistoryWithHDADetails(m,l,k).then(function(o,n){j.trigger("switched-history",j)})},createNewHistory:function(l){if(!Galaxy||!Galaxy.currUser||Galaxy.currUser.isAnonymous()){this.displayMessage("error",_l("You must be logged in to create histories"));return $.when()}var j=this,k=function(){return jQuery.post(galaxy_config.root+"api/histories",{current:true})};return this.loadHistory(undefined,l,k).then(function(n,m){j.trigger("new-history",j)})},loadHistoryWithHDADetails:function(m,l,k,o){var j=this,n=function(p){return j.getExpandedHdaIds(p.id)};return this.loadHistory(m,l,k,o,n)},loadHistory:function(m,l,k,p,n){this.trigger("loading-history",this);l=l||{};var j=this;var o=c.History.getHistoryData(m,{historyFn:k,hdaFn:p,hdaDetailIds:l.initiallyExpanded||n});return this._loadHistoryFromXHR(o,l).fail(function(s,q,r){j.trigger("error",j,s,l,_l("An error was encountered while "+q),{historyId:m,history:r||{}})}).always(function(){j.trigger("loading-done",j)})},_loadHistoryFromXHR:function(l,k){var j=this;l.then(function(m,n){j.setModel(m,n,k)});l.fail(function(n,m){j.render()});return l},setModel:function(l,j,k){k=k||{};if(this.model){this.model.clearUpdateTimeout();this.stopListening(this.model);this.stopListening(this.model.hdas)}this.hdaViews={};if(Galaxy&&Galaxy.currUser){l.user=Galaxy.currUser.toJSON()}this.model=new c.History(l,j,k);this._setUpWebStorage(k.initiallyExpanded,k.show_deleted,k.show_hidden);this._setUpModelEventHandlers();this.trigger("new-model",this);this.render();return this},_setUpWebStorage:function(k,j,l){this.storage=new d({id:d.historyStorageKey(this.model.get("id"))});if(_.isObject(k)){this.storage.set("exandedHdas",k)}if(_.isBoolean(j)){this.storage.set("show_deleted",j)}if(_.isBoolean(l)){this.storage.set("show_hidden",l)}this.trigger("new-storage",this.storage,this);this.log(this+" (init'd) storage:",this.storage.get())},clearWebStorage:function(){for(var j in sessionStorage){if(j.indexOf("history:")===0){sessionStorage.removeItem(j)}}},getStoredOptions:function(k){if(!k||k==="current"){return(this.storage)?(this.storage.get()):({})}var j=sessionStorage.getItem(d.historyStorageKey(k));return(j===null)?({}):(JSON.parse(j))},getExpandedHdaIds:function(j){var k=this.getStoredOptions(j).expandedHdas;return((_.isEmpty(k))?([]):(_.keys(k)))},_setUpModelEventHandlers:function(){this.model.on("error error:hdas",function(k,m,j,l){this.errorHandler(k,m,j,l)},this);this.model.on("change:nice_size",this.updateHistoryDiskSize,this);if(Galaxy&&Galaxy.quotaMeter){this.listenTo(this.model,"change:nice_size",function(){Galaxy.quotaMeter.update()})}this.model.hdas.on("add",this.addHdaView,this);this.model.hdas.on("change:deleted",this.handleHdaDeletionChange,this);this.model.hdas.on("change:visible",this.handleHdaVisibleChange,this);this.model.hdas.on("change:purged",function(j){this.model.fetch()},this);this.model.hdas.on("state:ready",function(k,l,j){if((!k.get("visible"))&&(!this.storage.get("show_hidden"))){this.removeHdaView(this.hdaViews[k.id])}},this)},render:function(l,m){l=(l===undefined)?(this.fxSpeed):(l);var j=this,k;if(this.model){k=this.renderModel()}else{k=this.renderWithoutModel()}$(j).queue("fx",[function(n){if(l&&j.$el.is(":visible")){j.$el.fadeOut(l,n)}else{n()}},function(n){j.$el.empty();if(k){j.$el.append(k.children());j.renderBasedOnPrefs()}n()},function(n){if(l&&!j.$el.is(":visible")){j.$el.fadeIn(l,n)}else{n()}},function(n){if(m){m.call(this)}j.trigger("rendered",this);n()}]);return this},renderWithoutModel:function(){var j=$("<div/>"),k=$("<div/>").addClass("message-container").css({"margin-left":"4px","margin-right":"4px"});return j.append(k)},renderModel:function(){var j=$("<div/>");if(!Galaxy||!Galaxy.currUser||Galaxy.currUser.isAnonymous()){j.append(e.templates.anonHistoryPanel(this.model.toJSON()))}else{j.append(e.templates.historyPanel(this.model.toJSON()));if(Galaxy.currUser.id&&Galaxy.currUser.id===this.model.get("user_id")){this._renderTags(j);this._renderAnnotation(j)}}j.find(".history-secondary-actions").prepend(this._renderSearchButton());this._setUpBehaviours(j);this.renderHdas(j);return j},renderBasedOnPrefs:function(){if(this.preferences.get("searching")){this.showSearchControls(0)}},_renderTags:function(j){var k=this;this.tagsEditor=new TagsEditor({model:this.model,el:j.find(".history-controls .tags-display"),onshowFirstTime:function(){this.render()},onshow:function(){k.preferences.set("tagsEditorShown",true);k.toggleHDATagEditors(true,k.fxSpeed)},onhide:function(){k.preferences.set("tagsEditorShown",false);k.toggleHDATagEditors(false,k.fxSpeed)},$activator:faIconButton({title:_l("Edit history tags"),classes:"history-tag-btn",faIcon:"fa-tags"}).appendTo(j.find(".history-secondary-actions"))});if(this.preferences.get("tagsEditorShown")){this.tagsEditor.toggle(true)}},_renderAnnotation:function(j){var k=this;this.annotationEditor=new AnnotationEditor({model:this.model,el:j.find(".history-controls .annotation-display"),onshowFirstTime:function(){this.render()},onshow:function(){k.preferences.set("annotationEditorShown",true);k.toggleHDAAnnotationEditors(true,k.fxSpeed)},onhide:function(){k.preferences.set("annotationEditorShown",false);k.toggleHDAAnnotationEditors(false,k.fxSpeed)},$activator:faIconButton({title:_l("Edit history Annotation"),classes:"history-annotate-btn",faIcon:"fa-comment"}).appendTo(j.find(".history-secondary-actions"))});if(this.preferences.get("annotationEditorShown")){this.annotationEditor.toggle(true)}},_renderSearchButton:function(j){return faIconButton({title:_l("Search datasets"),classes:"history-search-btn",faIcon:"fa-search"})},_renderSelectButton:function(j){return faIconButton({title:_l("Operations on multiple datasets"),classes:"history-select-btn",faIcon:"fa-check-square-o"})},_setUpBehaviours:function(j){j=j||this.$el;j.find("[title]").tooltip({placement:"bottom"});if((!this.model)||(!Galaxy.currUser||Galaxy.currUser.isAnonymous())||(Galaxy.currUser.id!==this.model.get("user_id"))){return}var k=this;j.find(".history-name").attr("title",_l("Click to rename history")).tooltip({placement:"bottom"}).make_text_editable({on_finish:function(l){var m=k.model.get("name");if(l&&l!==m){k.$el.find(".history-name").text(l);k.model.save({name:l}).fail(function(){k.$el.find(".history-name").text(k.model.previous("name"))})}else{k.$el.find(".history-name").text(m)}}});this._setUpDatasetActionsPopup(j)},_setUpDatasetActionsPopup:function(j){var k=this;(new PopupMenu(j.find(".history-dataset-action-popup-btn"),[{html:_l("Hide datasets"),func:function(){var l=f.HistoryDatasetAssociation.prototype.hide;k.getSelectedHdaCollection().ajaxQueue(l)}},{html:_l("Unhide datasets"),func:function(){var l=f.HistoryDatasetAssociation.prototype.unhide;k.getSelectedHdaCollection().ajaxQueue(l)}},{html:_l("Delete datasets"),func:function(){var l=f.HistoryDatasetAssociation.prototype["delete"];k.getSelectedHdaCollection().ajaxQueue(l)}},{html:_l("Undelete datasets"),func:function(){var l=f.HistoryDatasetAssociation.prototype.undelete;k.getSelectedHdaCollection().ajaxQueue(l)}},{html:_l("Permanently delete datasets"),func:function(){if(confirm(_l("This will permanently remove the data in your datasets. Are you sure?"))){var l=f.HistoryDatasetAssociation.prototype.purge;k.getSelectedHdaCollection().ajaxQueue(l)}}}]))},refreshHdas:function(k,j){if(this.model){return this.model.refresh(k,j)}return $.when()},addHdaView:function(m){this.log("add."+this,m);var k=this;if(!m.isVisible(this.storage.get("show_deleted"),this.storage.get("show_hidden"))){return}$({}).queue([function l(o){var n=k.$el.find(k.emptyMsgSelector);if(n.is(":visible")){n.fadeOut(k.fxSpeed,o)}else{o()}},function j(o){k.scrollToTop();var n=k.$el.find(k.datasetsSelector);k.createHdaView(m).$el.hide().prependTo(n).slideDown(k.fxSpeed)}])},createHdaView:function(l){var k=l.get("id"),j=this.storage.get("expandedHdas")[k],m=new this.HDAViewClass({model:l,linkTarget:this.linkTarget,expanded:j,tagsEditorShown:this.preferences.get("tagsEditorShown"),annotationEditorShown:this.preferences.get("annotationEditorShown"),selectable:this.selecting,hasUser:this.model.ownedByCurrUser(),logger:this.logger});this._setUpHdaListeners(m);this.hdaViews[k]=m;return m.render()},_setUpHdaListeners:function(k){var j=this;k.on("body-expanded",function(l){j.storage.addExpandedHda(l)});k.on("body-collapsed",function(l){j.storage.removeExpandedHda(l)});k.on("error",function(m,o,l,n){j.errorHandler(m,o,l,n)})},handleHdaDeletionChange:function(j){if(j.get("deleted")&&!this.storage.get("show_deleted")){this.removeHdaView(this.hdaViews[j.id])}},handleHdaVisibleChange:function(j){if(j.hidden()&&!this.storage.get("show_hidden")){this.removeHdaView(this.hdaViews[j.id])}},removeHdaView:function(k){if(!k){return}var j=this;k.$el.fadeOut(j.fxSpeed,function(){k.off();k.remove();delete j.hdaViews[k.model.id];if(_.isEmpty(j.hdaViews)){j.$el.find(j.emptyMsgSelector).fadeIn(j.fxSpeed,function(){j.trigger("empty-history",j)})}})},renderHdas:function(l){l=l||this.$el;this.hdaViews={};var k=this,j=l.find(this.datasetsSelector),m=this.model.hdas.getVisible(this.storage.get("show_deleted"),this.storage.get("show_hidden"),this.filters);j.empty();if(m.length){m.each(function(n){j.prepend(k.createHdaView(n).$el)});l.find(this.emptyMsgSelector).hide()}else{l.find(this.emptyMsgSelector).show()}return this.hdaViews},toggleHDATagEditors:function(j){var k=arguments;_.each(this.hdaViews,function(l){if(l.tagsEditor){l.tagsEditor.toggle.apply(l.tagsEditor,k)}})},toggleHDAAnnotationEditors:function(j){var k=arguments;_.each(this.hdaViews,function(l){if(l.annotationEditor){l.annotationEditor.toggle.apply(l.annotationEditor,k)}})},events:{"click .message-container":"clearMessages","click .history-search-btn":"toggleSearchControls","click .history-select-btn":function(j){this.toggleSelectors(this.fxSpeed)},"click .history-select-all-datasets-btn":"selectAllDatasets"},updateHistoryDiskSize:function(){this.$el.find(".history-size").text(this.model.get("nice_size"))},collapseAllHdaBodies:function(){_.each(this.hdaViews,function(j){j.toggleBodyVisibility(null,false)});this.storage.set("expandedHdas",{})},toggleShowDeleted:function(){this.storage.set("show_deleted",!this.storage.get("show_deleted"));this.renderHdas();return this.storage.get("show_deleted")},toggleShowHidden:function(){this.storage.set("show_hidden",!this.storage.get("show_hidden"));this.renderHdas();return this.storage.get("show_hidden")},renderSearchControls:function(k){var l=this;function n(o){l.searchFor=o;l.filters=[function(p){return p.matchesAll(l.searchFor)}];l.trigger("search:searching",o,l);l.renderHdas()}function j(o){if(l.model.hdas.haveDetails()){n(o);return}l.$el.find(".history-search-controls").searchInput("toggle-loading");l.model.hdas.fetchAllDetails({silent:true}).always(function(){l.$el.find(".history-search-controls").searchInput("toggle-loading")}).done(function(){n(o)})}function m(){l.searchFor="";l.filters=[];l.trigger("search:clear",l);l.renderHdas()}return k.searchInput({initialVal:l.searchFor,name:"history-search",placeholder:"search datasets",classes:"history-search",onfirstsearch:j,onsearch:n,onclear:m})},showSearchControls:function(l){l=(l===undefined)?(this.fxSpeed):(l);var j=this,k=this.$el.find(".history-search-controls");if(!k.children().size()){k=this.renderSearchControls(k).hide()}k.show(l,function(){$(this).find("input").focus();j.preferences.set("searching",true)})},hideSearchControls:function(){speed=(speed===undefined)?(this.fxSpeed):(speed);var j=this;this.$el.find(".history-search-controls").hide(speed,function(){j.preferences.set("searching",false)})},toggleSearchControls:function(j){speed=(jQuery.type(j)==="number")?(j):(this.fxSpeed);if(this.$el.find(".history-search-controls").is(":visible")){this.hideSearchControls(speed)}else{this.showSearchControls(speed)}},showSelectors:function(j){this.selecting=true;this.$el.find(".history-dataset-actions").slideDown(j);_.each(this.hdaViews,function(k){k.showSelector(j)})},hideSelectors:function(j){this.selecting=false;this.$el.find(".history-dataset-actions").slideUp(j);_.each(this.hdaViews,function(k){k.hideSelector(j)})},toggleSelectors:function(j){if(!this.selecting){this.showSelectors(j)}else{this.hideSelectors(j)}},selectAllDatasets:function(k){var j=this.$el.find(".history-select-all-datasets-btn");currMode=j.data("mode");if(currMode==="select"){_.each(this.hdaViews,function(l){l.select(k)});j.data("mode","deselect");j.text(_l("De-select all"))}else{if(currMode==="deselect"){_.each(this.hdaViews,function(l){l.deselect(k)});j.data("mode","select");j.text(_l("Select all"))}}},getSelectedHdaViews:function(){return _.filter(this.hdaViews,function(j){return j.selected})},getSelectedHdaCollection:function(){return new f.HDACollection(_.map(this.getSelectedHdaViews(),function(j){return j.model}),{historyId:this.model.id})},showLoadingIndicator:function(k,j,l){j=(j!==undefined)?(j):(this.fxSpeed);if(!this.indicator){this.indicator=new LoadingIndicator(this.$el,this.$el.parent())}if(!this.$el.is(":visible")){this.indicator.show(0,l)}else{this.$el.fadeOut(j);this.indicator.show(k,j,l)}},hideLoadingIndicator:function(j,k){j=(j!==undefined)?(j):(this.fxSpeed);if(this.indicator){this.indicator.hide(j,k)}},displayMessage:function(o,p,n){var l=this;this.scrollToTop();var m=this.$el.find(this.msgsSelector),j=$("<div/>").addClass(o+"message").html(p);if(!_.isEmpty(n)){var k=$('<a href="javascript:void(0)">Details</a>').click(function(){Galaxy.modal.show(l.messageToModalOptions(o,p,n));return false});j.append(" ",k)}return m.html(j)},messageToModalOptions:function(n,p,m){var j=this,o=$("<div/>"),l={title:"Details"};function k(q){q=_.omit(q,_.functions(q));return["<table>",_.map(q,function(s,r){s=(_.isObject(s))?(k(s)):(s);return'<tr><td style="vertical-align: top; color: grey">'+r+'</td><td style="padding-left: 8px">'+s+"</td></tr>"}).join(""),"</table>"].join("")}if(_.isObject(m)){l.body=o.append(k(m))}else{l.body=o.html(m)}l.buttons={Ok:function(){Galaxy.modal.hide();j.clearMessages()}};return l},clearMessages:function(){var j=this.$el.find(this.msgsSelector);j.empty()},scrollPosition:function(){return this.$el.parent().scrollTop()},scrollTo:function(j){this.$el.parent().scrollTop(j)},scrollToTop:function(){this.$el.parent().scrollTop(0);return this},scrollIntoView:function(k,l){if(!l){this.$el.parent().parent().scrollTop(k);return this}var j=window,m=this.$el.parent().parent(),o=$(j).innerHeight(),n=(o/2)-(l/2);$(m).scrollTop(k-n);return this},scrollToId:function(k){if((!k)||(!this.hdaViews[k])){return this}var j=this.hdaViews[k].$el;this.scrollIntoView(j.offset().top,j.outerHeight());return this},scrollToHid:function(j){var k=this.model.hdas.getByHid(j);if(!k){return this}return this.scrollToId(k.id)},connectToQuotaMeter:function(j){if(!j){return this}this.listenTo(j,"quota:over",this.showQuotaMessage);this.listenTo(j,"quota:under",this.hideQuotaMessage);this.on("rendered rendered:initial",function(){if(j&&j.isOverQuota()){this.showQuotaMessage()}});return this},showQuotaMessage:function(){var j=this.$el.find(".quota-message");if(j.is(":hidden")){j.slideDown(this.fxSpeed)}},hideQuotaMessage:function(){var j=this.$el.find(".quota-message");if(!j.is(":hidden")){j.slideUp(this.fxSpeed)}},connectToOptionsMenu:function(j){if(!j){return this}this.on("new-storage",function(l,k){if(j&&l){j.findItemByHtml(_l("Include Deleted Datasets")).checked=l.get("show_deleted");j.findItemByHtml(_l("Include Hidden Datasets")).checked=l.get("show_hidden")}});return this},toString:function(){return"HistoryPanel("+((this.model)?(this.model.get("name")):(""))+")"}});e.templates={historyPanel:Handlebars.templates["template-history-historyPanel"],anonHistoryPanel:Handlebars.templates["template-history-historyPanel-anon"]};return{HistoryPanel:e}});
\ No newline at end of file
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0
                            
                          
                          
                            
    
                          
                        
                    
                        
                            
                                
                            
                            commit/galaxy-central: greg: Fix for determining a guid in the install_manager.
                        
                        
by commits-noreply@bitbucket.org 19 Dec '13
                    by commits-noreply@bitbucket.org 19 Dec '13
19 Dec '13
                    
                        1 new commit in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/a9a0ac9c1afa/
Changeset:   a9a0ac9c1afa
Branch:      stable
User:        greg
Date:        2013-12-19 01:03:05
Summary:     Fix for determining a guid in the install_manager.
Affected #:  1 file
diff -r 26da838b778c68a5577d4cca9836f6ef70b8ed1b -r a9a0ac9c1afae8dd0d662c9c800d0228c883c107 lib/tool_shed/galaxy_install/install_manager.py
--- a/lib/tool_shed/galaxy_install/install_manager.py
+++ b/lib/tool_shed/galaxy_install/install_manager.py
@@ -244,21 +244,20 @@
 
     def get_guid( self, repository_clone_url, relative_install_dir, tool_config ):
         if self.shed_config_dict.get( 'tool_path' ):
-            relative_install_dir = os.path.join( self.shed_config_dict['tool_path'], relative_install_dir )
-        found = False
+            relative_install_dir = os.path.join( self.shed_config_dict[ 'tool_path' ], relative_install_dir )
+        tool_config_filename = suc.strip_path( tool_config )
         for root, dirs, files in os.walk( relative_install_dir ):
             if root.find( '.hg' ) < 0 and root.find( 'hgrc' ) < 0:
                 if '.hg' in dirs:
                     dirs.remove( '.hg' )
                 for name in files:
-                    if name == tool_config:
-                        found = True
-                        break
-            if found:
-                break
-        full_path = str( os.path.abspath( os.path.join( root, name ) ) )
-        tool = self.toolbox.load_tool( full_path )
-        return suc.generate_tool_guid( repository_clone_url, tool )
+                    filename = suc.strip_path( name )
+                    if filename == tool_config_filename:
+                        full_path = str( os.path.abspath( os.path.join( root, name ) ) )
+                        tool = self.toolbox.load_tool( full_path )
+                        return suc.generate_tool_guid( repository_clone_url, tool )
+        # Not quite sure what should happen here, throw an exception or what?
+        return None
 
     def get_prior_install_required_dict( self, tool_shed_repositories, repository_dependencies_dict ):
         """
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0
                            
                          
                          
                            
    
                          
                        
                    
                        
                            
                                
                            
                            commit/galaxy-central: greg: Don't uninstall repositories when testing tool dependency definitions in the tool shed's install and test process.
                        
                        
by commits-noreply@bitbucket.org 19 Dec '13
                    by commits-noreply@bitbucket.org 19 Dec '13
19 Dec '13
                    
                        1 new commit in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/67fe4b091f0a/
Changeset:   67fe4b091f0a
User:        greg
Date:        2013-12-19 14:47:20
Summary:     Don't uninstall repositories when testing tool dependency definitions in the tool shed's install and test process.
Affected #:  1 file
diff -r 5bfa1d47b3999a695a04294adb13da0b5b3ec041 -r 67fe4b091f0ad66d68d06a36b8fb645aa445d032 test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
--- a/test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
+++ b/test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
@@ -201,28 +201,18 @@
                 else:
                     log.debug( 'Installation succeeded for revision %s of repository %s owned by %s.' % \
                         ( changeset_revision, name, owner ) )
-                    # Keep statistics for this repository's tool dependencies that resulted in installation errors.
-                    for missing_tool_dependency in repository.missing_tool_dependencies:
-                        name = str( missing_tool_dependency.name )
-                        type = str( missing_tool_dependency.type )
-                        version = str( missing_tool_dependency.version )
-                        error_message = unicodify( missing_tool_dependency.error_message )
-                        missing_tool_dependency_info_dict = dict( type=type,
-                                                                  name=name,
-                                                                  version=version,
-                                                                  error_message=error_message )
-                        install_and_test_statistics_dict[ 'tool_dependencies_with_installation_error' ].append( missing_tool_dependency_info_dict )
-                    # Attempt to uninstall this repository  and all of its dependencies if its repository dependencies or
-                    # tool dependencies resulted in an installation error.
-                    missing_tool_dependencies = install_and_test_base_util.get_missing_tool_dependencies( repository )
-                    if missing_tool_dependencies or repository.missing_repository_dependencies:
-                        install_and_test_base_util.handle_missing_dependencies( app=app,
-                                                                                repository=repository,
-                                                                                missing_tool_dependencies=missing_tool_dependencies,
-                                                                                repository_dict=repository_dict,
-                                                                                tool_test_results_dicts=tool_test_results_dicts,
-                                                                                tool_test_results_dict=tool_test_results_dict,
-                                                                                can_update_tool_shed=can_update_tool_shed )
+                    if repository.missing_tool_dependencies:
+                        # Keep statistics for this repository's tool dependencies that resulted in installation errors.
+                        for missing_tool_dependency in repository.missing_tool_dependencies:
+                            name = str( missing_tool_dependency.name )
+                            type = str( missing_tool_dependency.type )
+                            version = str( missing_tool_dependency.version )
+                            error_message = unicodify( missing_tool_dependency.error_message )
+                            missing_tool_dependency_info_dict = dict( type=type,
+                                                                      name=name,
+                                                                      version=version,
+                                                                      error_message=error_message )
+                            install_and_test_statistics_dict[ 'tool_dependencies_with_installation_error' ].append( missing_tool_dependency_info_dict )
                     else:
                         # This repository and all of its dependencies were successfully installed.
                         install_and_test_statistics_dict[ 'successful_installations' ].append( repository_identifier_dict )
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0
                            
                          
                          
                            
    
                          
                        
                    
                        
                            
                                
                            
                            commit/galaxy-central: greg: Fix for handling missing dependencies in the tool shed's install and test framework.
                        
                        
by commits-noreply@bitbucket.org 19 Dec '13
                    by commits-noreply@bitbucket.org 19 Dec '13
19 Dec '13
                    
                        1 new commit in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/5bfa1d47b399/
Changeset:   5bfa1d47b399
User:        greg
Date:        2013-12-19 02:32:15
Summary:     Fix for handling missing dependencies in the tool shed's install and test framework.
Affected #:  2 files
diff -r 4077a5a1f801a772acd01d19c9695329a42c5958 -r 5bfa1d47b3999a695a04294adb13da0b5b3ec041 test/install_and_test_tool_shed_repositories/repositories_with_tools/functional_tests.py
--- a/test/install_and_test_tool_shed_repositories/repositories_with_tools/functional_tests.py
+++ b/test/install_and_test_tool_shed_repositories/repositories_with_tools/functional_tests.py
@@ -286,16 +286,15 @@
                     if 'missing_test_components' not in tool_test_results_dict:
                         tool_test_results_dict[ 'missing_test_components' ] = []
                     missing_tool_dependencies = install_and_test_base_util.get_missing_tool_dependencies( repository )
-                    if missing_tool_dependencies or repository.missing_repository_dependencies:
+                    if missing_tool_dependencies or repository.missing_repository_dependencies:                        
                         install_and_test_statistics_dict = \
-                            install_and_test_base_util.handle_missing_dependencies( app,
-                                                                                    repository,
-                                                                                    missing_tool_dependencies,
-                                                                                    repository_dict,
-                                                                                    tool_test_results_dicts,
-                                                                                    tool_test_results_dict,
-                                                                                    install_and_test_statistics_dict,
-                                                                                    can_update_tool_shed )
+                            install_and_test_base_util.handle_missing_dependencies( app=app,
+                                                                                    repository=repository,
+                                                                                    missing_tool_dependencies=missing_tool_dependencies,
+                                                                                    repository_dict=repository_dict,
+                                                                                    tool_test_results_dicts=tool_test_results_dicts,
+                                                                                    tool_test_results_dict=tool_test_results_dict,
+                                                                                    can_update_tool_shed=can_update_tool_shed )
                         # Set the test_toolbox.toolbox module-level variable to the new app.toolbox.
                         test_toolbox.toolbox = app.toolbox
                     else:
diff -r 4077a5a1f801a772acd01d19c9695329a42c5958 -r 5bfa1d47b3999a695a04294adb13da0b5b3ec041 test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
--- a/test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
+++ b/test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
@@ -216,14 +216,13 @@
                     # tool dependencies resulted in an installation error.
                     missing_tool_dependencies = install_and_test_base_util.get_missing_tool_dependencies( repository )
                     if missing_tool_dependencies or repository.missing_repository_dependencies:
-                        install_and_test_base_util.handle_missing_dependencies( app,
-                                                                                repository,
-                                                                                missing_tool_dependencies,
-                                                                                repository_dict,
-                                                                                tool_test_results_dicts,
-                                                                                tool_test_results_dict,
-                                                                                install_and_test_statistics_dict,
-                                                                                can_update_tool_shed )
+                        install_and_test_base_util.handle_missing_dependencies( app=app,
+                                                                                repository=repository,
+                                                                                missing_tool_dependencies=missing_tool_dependencies,
+                                                                                repository_dict=repository_dict,
+                                                                                tool_test_results_dicts=tool_test_results_dicts,
+                                                                                tool_test_results_dict=tool_test_results_dict,
+                                                                                can_update_tool_shed=can_update_tool_shed )
                     else:
                         # This repository and all of its dependencies were successfully installed.
                         install_and_test_statistics_dict[ 'successful_installations' ].append( repository_identifier_dict )
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0
                            
                          
                          
                            
    
                          
                        
                    
                        
                            
                                
                            
                            commit/galaxy-central: greg: Fix imports in the tool shed's functional test framework.
                        
                        
by commits-noreply@bitbucket.org 19 Dec '13
                    by commits-noreply@bitbucket.org 19 Dec '13
19 Dec '13
                    
                        1 new commit in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/4077a5a1f801/
Changeset:   4077a5a1f801
User:        greg
Date:        2013-12-19 01:09:11
Summary:     Fix imports in the tool shed's functional test framework.
Affected #:  3 files
diff -r a53f2f700699c9d88b1896814b3eb2ec8cc028b8 -r 4077a5a1f801a772acd01d19c9695329a42c5958 test/install_and_test_tool_shed_repositories/base/util.py
--- a/test/install_and_test_tool_shed_repositories/base/util.py
+++ b/test/install_and_test_tool_shed_repositories/base/util.py
@@ -595,7 +595,6 @@
         exclude_list.append( exclude_dict )
     log.debug( 'The xml document %s containing the exclude list defines the following %s repositories to be excluded from testing...' % \
         ( str( xml_filename ), str( exclude_count ) ) )
-    #if '-list_repositories' in sys.argv:
     for name, owner, changeset_revision in exclude_verbose:
         if changeset_revision:
             log.debug( 'Repository %s owned by %s, changeset revision %s.' % ( str( name ), str( owner ), str( changeset_revision ) ) )
diff -r a53f2f700699c9d88b1896814b3eb2ec8cc028b8 -r 4077a5a1f801a772acd01d19c9695329a42c5958 test/install_and_test_tool_shed_repositories/repositories_with_tools/functional_tests.py
--- a/test/install_and_test_tool_shed_repositories/repositories_with_tools/functional_tests.py
+++ b/test/install_and_test_tool_shed_repositories/repositories_with_tools/functional_tests.py
@@ -45,6 +45,7 @@
 from galaxy.app import UniverseApplication
 from galaxy.util.json import from_json_string
 from galaxy.util.json import to_json_string
+from galaxy.util import unicodify
 from galaxy.web import buildapp
 from functional_tests import generate_config_file
 from nose.plugins import Plugin
diff -r a53f2f700699c9d88b1896814b3eb2ec8cc028b8 -r 4077a5a1f801a772acd01d19c9695329a42c5958 test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
--- a/test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
+++ b/test/install_and_test_tool_shed_repositories/tool_dependency_definitions/functional_tests.py
@@ -45,6 +45,7 @@
 from galaxy.app import UniverseApplication
 from galaxy.util.json import from_json_string
 from galaxy.util.json import to_json_string
+from galaxy.util import unicodify
 from galaxy.web import buildapp
 from functional_tests import generate_config_file
 from nose.plugins import Plugin
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0
                            
                          
                          
                            
    
                          
                        
                    
                        
                            
                                
                            
                            commit/galaxy-central: greg: Fix for determining a guid in the install_manager.
                        
                        
by commits-noreply@bitbucket.org 19 Dec '13
                    by commits-noreply@bitbucket.org 19 Dec '13
19 Dec '13
                    
                        1 new commit in galaxy-central:
https://bitbucket.org/galaxy/galaxy-central/commits/a53f2f700699/
Changeset:   a53f2f700699
User:        greg
Date:        2013-12-19 01:03:05
Summary:     Fix for determining a guid in the install_manager.
Affected #:  1 file
diff -r 9fa754a203b7312d58211b5e1e7140bf36aec76a -r a53f2f700699c9d88b1896814b3eb2ec8cc028b8 lib/tool_shed/galaxy_install/install_manager.py
--- a/lib/tool_shed/galaxy_install/install_manager.py
+++ b/lib/tool_shed/galaxy_install/install_manager.py
@@ -244,21 +244,20 @@
 
     def get_guid( self, repository_clone_url, relative_install_dir, tool_config ):
         if self.shed_config_dict.get( 'tool_path' ):
-            relative_install_dir = os.path.join( self.shed_config_dict['tool_path'], relative_install_dir )
-        found = False
+            relative_install_dir = os.path.join( self.shed_config_dict[ 'tool_path' ], relative_install_dir )
+        tool_config_filename = suc.strip_path( tool_config )
         for root, dirs, files in os.walk( relative_install_dir ):
             if root.find( '.hg' ) < 0 and root.find( 'hgrc' ) < 0:
                 if '.hg' in dirs:
                     dirs.remove( '.hg' )
                 for name in files:
-                    if name == tool_config:
-                        found = True
-                        break
-            if found:
-                break
-        full_path = str( os.path.abspath( os.path.join( root, name ) ) )
-        tool = self.toolbox.load_tool( full_path )
-        return suc.generate_tool_guid( repository_clone_url, tool )
+                    filename = suc.strip_path( name )
+                    if filename == tool_config_filename:
+                        full_path = str( os.path.abspath( os.path.join( root, name ) ) )
+                        tool = self.toolbox.load_tool( full_path )
+                        return suc.generate_tool_guid( repository_clone_url, tool )
+        # Not quite sure what should happen here, throw an exception or what?
+        return None
 
     def get_prior_install_required_dict( self, tool_shed_repositories, repository_dependencies_dict ):
         """
Repository URL: https://bitbucket.org/galaxy/galaxy-central/
--
This is a commit notification from bitbucket.org. You are receiving
this because you have the service enabled, addressing the recipient of
this email.
                    
                  
                  
                          
                            
                            1
                            
                          
                          
                            
                            0