OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
openproject/lib/open_project/storage.rb

105 lines
3.1 KiB

#-- encoding: UTF-8
#-- copyright
# OpenProject is a project management system.
# Copyright (C) 2012-2017 the OpenProject Foundation (OPF)
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 3.
#
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
# Copyright (C) 2006-2017 Jean-Philippe Lang
# Copyright (C) 2010-2013 the ChiliProject Team
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# See doc/COPYRIGHT.rdoc for more details.
#++
require 'sys/filesystem'
module OpenProject
module Storage
class << self
##
# List available storage endpoints
def mount_information
list_mounts(known_storage_paths)
end
##
# List available storage paths of OpenProject
def known_storage_paths
paths = {}
# SCM vendors
OpenProject::Scm::Manager.managed_paths.each do |vendor, path|
paths[vendor] = {
path: path,
label: I18n.t(:label_managed_repositories_vendor, vendor: vendor.to_s.camelize)
}
end
# Attachments
paths[:attachments] = {
path: OpenProject::Configuration.attachments_storage_path.to_s,
label: I18n.t('attributes.attachments')
}
paths
end
private
##
# Return mount information based on their filesystem id
#
def list_mounts(entries)
mounts = {}
entries.each do |_identifier, entry|
stat = read_fs_info(entry[:path])
next if stat.nil?
# Aggregate directories by filesystem, so we don't return
# storage information from the same filesystem twice
fs_id = stat[:id]
if mounts[fs_id].nil?
mounts[fs_id] = { labels: [entry[:label]], data: stat }
else
mounts[fs_id][:labels] << entry[:label]
end
end
mounts
end
def read_fs_info(dir)
stat = Sys::Filesystem.stat(dir)
{
dir: dir,
free: stat.bytes_free,
used: stat.bytes_used,
percent_used: stat.percent_used,
total: stat.bytes_total,
id: stat.filesystem_id
}
rescue SystemCallError, Sys::Filesystem::Error => e
Rails.logger.warn("Can't read storage information on #{dir}: #{e.message}")
nil
end
end
end
end