mirror of
https://github.com/fog/fog.git
synced 2022-11-09 13:51:43 -05:00
2e0b7e545a
Done with `rubocop --auto-correct --only EmptyLineBetweenDefs,EmptyLines,EmptyLinesAroundBody`
305 lines
10 KiB
Ruby
305 lines
10 KiB
Ruby
require 'fog/core/model'
|
|
require 'fog/hp/models/storage/files'
|
|
require 'fog/hp/models/storage/metadata'
|
|
|
|
module Fog
|
|
module Storage
|
|
class HP
|
|
class Directory < Fog::Model
|
|
identity :key, :aliases => 'name'
|
|
|
|
attribute :bytes, :aliases => 'X-Container-Bytes-Used'
|
|
attribute :count, :aliases => 'X-Container-Object-Count'
|
|
attribute :sync_to, :aliases => 'X-Container-Sync-To'
|
|
attribute :sync_key, :aliases => 'X-Container-Sync-Key'
|
|
attribute :web_index, :aliases => 'X-Container-Meta-Web-Index'
|
|
attribute :web_listings, :aliases => 'X-Container-Meta-Web-Listings'
|
|
attribute :web_listings_css, :aliases => 'X-Container-Meta-Web-Listings-Css'
|
|
attribute :web_error, :aliases => 'X-Container-Meta-Web-Error'
|
|
|
|
def initialize(attributes = {})
|
|
@read_acl = []
|
|
@write_acl = []
|
|
super
|
|
end
|
|
|
|
def read_acl
|
|
@read_acl || []
|
|
end
|
|
|
|
def write_acl
|
|
@write_acl || []
|
|
end
|
|
|
|
def can_read?(user)
|
|
return false if @read_acl.nil?
|
|
list_users_with_read.include?(user)
|
|
end
|
|
|
|
def can_write?(user)
|
|
return false if @write_acl.nil?
|
|
list_users_with_write.include?(user)
|
|
end
|
|
|
|
def can_read_write?(user)
|
|
can_read?(user) && can_write?(user)
|
|
end
|
|
|
|
def list_users_with_read
|
|
users = []
|
|
users = @read_acl.map {|acl| acl.split(':')[1]} unless @read_acl.nil?
|
|
return users
|
|
end
|
|
|
|
def list_users_with_write
|
|
users = []
|
|
users = @write_acl.map {|acl| acl.split(':')[1]} unless @write_acl.nil?
|
|
return users
|
|
end
|
|
|
|
def grant(perm, users=nil)
|
|
# support passing in a list of users in a comma-separated list or as an Array
|
|
if users.is_a?(String)
|
|
user_list = users.split(',')
|
|
else
|
|
user_list = users
|
|
end
|
|
r_acl, w_acl = service.perm_to_acl(perm, user_list)
|
|
unless r_acl.nil? || r_acl.empty?
|
|
@read_acl = [] if @read_acl.nil?
|
|
@read_acl = @read_acl + r_acl
|
|
@read_acl.uniq!
|
|
end
|
|
unless w_acl.nil? || w_acl.empty?
|
|
@write_acl = [] if @write_acl.nil?
|
|
@write_acl = @write_acl + w_acl
|
|
@write_acl.uniq!
|
|
end
|
|
true
|
|
end
|
|
|
|
def revoke(perm, users=nil)
|
|
# support passing in a list of users in a comma-separated list or as an Array
|
|
if users.is_a?(String)
|
|
user_list = users.split(',')
|
|
else
|
|
user_list = users
|
|
end
|
|
r_acl, w_acl = service.perm_to_acl(perm, user_list)
|
|
unless r_acl.nil? || r_acl.empty?
|
|
@read_acl = [] if @read_acl.nil?
|
|
@read_acl = @read_acl - r_acl
|
|
@read_acl.uniq!
|
|
end
|
|
unless w_acl.nil? || w_acl.empty?
|
|
@write_acl = [] if @write_acl.nil?
|
|
@write_acl = @write_acl - w_acl
|
|
@write_acl.uniq!
|
|
end
|
|
true
|
|
end
|
|
|
|
def metadata
|
|
@metadata ||= begin
|
|
Fog::Storage::HP::Metadata.new({
|
|
:service => service,
|
|
:parent => self
|
|
})
|
|
end
|
|
end
|
|
|
|
def metadata=(new_metadata={})
|
|
metas = []
|
|
new_metadata.each_pair {|k,v| metas << {'key' => k, 'value' => v} }
|
|
metadata.load(metas)
|
|
end
|
|
|
|
def destroy
|
|
requires :key
|
|
service.delete_container(key)
|
|
# If CDN service is available, try to delete the container if it was CDN-enabled
|
|
if cdn_enabled?
|
|
begin
|
|
service.cdn.delete_container(key)
|
|
rescue Fog::CDN::HP::NotFound
|
|
# ignore if cdn container not found
|
|
end
|
|
end
|
|
true
|
|
rescue Excon::Errors::NotFound, Fog::Storage::HP::NotFound
|
|
false
|
|
end
|
|
|
|
def files
|
|
@files ||= begin
|
|
Fog::Storage::HP::Files.new(
|
|
:directory => self,
|
|
:service => service
|
|
)
|
|
end
|
|
end
|
|
|
|
def public=(new_public)
|
|
if new_public
|
|
self.grant("pr")
|
|
else
|
|
self.revoke("pr")
|
|
end
|
|
@public = new_public
|
|
end
|
|
|
|
def public?
|
|
@read_acl = [] if @read_acl.nil?
|
|
if @read_acl.include?(".r:*")
|
|
true
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
def public_url
|
|
requires :key
|
|
@public_url ||= begin
|
|
begin response = service.head_container(key)
|
|
# escape the key to cover for special char. in container names
|
|
url = service.public_url(key)
|
|
rescue Fog::Storage::HP::NotFound => err
|
|
nil
|
|
end
|
|
end
|
|
end
|
|
|
|
def cdn_enable=(new_cdn_enable)
|
|
@cdn_enable ||= false
|
|
if (!service.cdn.nil? && service.cdn.enabled?)
|
|
@cdn_enable = new_cdn_enable
|
|
else
|
|
# since cdn service is not activated, container cannot be cdn-enabled
|
|
@cdn_enable = false
|
|
end
|
|
end
|
|
|
|
def cdn_enabled?
|
|
if (!service.cdn.nil? && service.cdn.enabled?)
|
|
begin response = service.cdn.head_container(key)
|
|
cdn_header = response.headers.fetch('X-Cdn-Enabled', nil)
|
|
if (!cdn_header.nil? && cdn_header == 'True')
|
|
@cdn_enable = true
|
|
else
|
|
@cdn_enable = false
|
|
end
|
|
# If CDN endpoint is unreachable, a SocketError is raised
|
|
rescue Fog::CDN::HP::NotFound, Excon::Errors::SocketError
|
|
@cdn_enable = false
|
|
end
|
|
else
|
|
@cdn_enable = false
|
|
end
|
|
end
|
|
|
|
def cdn_public_url
|
|
requires :key
|
|
@cdn_public_url ||= begin
|
|
# return the CDN public url from the appropriate uri from the header
|
|
begin response = service.cdn.head_container(key)
|
|
if response.headers['X-Cdn-Enabled'] == 'True'
|
|
response.headers.fetch('X-Cdn-Uri', nil)
|
|
end
|
|
rescue Fog::CDN::HP::NotFound
|
|
nil
|
|
end
|
|
end
|
|
end
|
|
|
|
def cdn_public_ssl_url
|
|
requires :key
|
|
@cdn_public_ssl_url ||= begin
|
|
# return the CDN public ssl url from the appropriate uri from the header
|
|
begin response = service.cdn.head_container(key)
|
|
if response.headers['X-Cdn-Enabled'] == 'True'
|
|
response.headers.fetch('X-Cdn-Ssl-Uri', nil)
|
|
end
|
|
rescue Fog::CDN::HP::NotFound
|
|
nil
|
|
end
|
|
end
|
|
end
|
|
|
|
def sync(target_dir, secret)
|
|
requires :key
|
|
# do not sync if dir is same as target dir
|
|
return false if target_dir.key == key
|
|
begin service.head_container(key)
|
|
if !target_dir.nil? && target_dir.is_a?(Fog::Storage::HP::Directory) && target_dir.respond_to?(:public_url) && !target_dir.public_url.nil?
|
|
# set sync metadata on source dir
|
|
self.sync_to = target_dir.public_url
|
|
self.sync_key = secret
|
|
# set sync metadata on target dir
|
|
target_dir.sync_key = secret
|
|
target_dir.save
|
|
true
|
|
else
|
|
false
|
|
end
|
|
rescue Fog::Storage::HP::NotFound
|
|
false
|
|
end
|
|
end
|
|
|
|
def save(new_options = {})
|
|
requires :key
|
|
options = {}
|
|
#these are default/previous options
|
|
meta_hash = {}
|
|
metadata.each { |meta| meta_hash.store(meta.key, meta.value) }
|
|
if meta_hash
|
|
options.merge!(meta_hash)
|
|
end
|
|
# write out the acls into the headers before save
|
|
options.merge!(service.perm_acl_to_header(@read_acl, @write_acl))
|
|
options.merge!({'X-Container-Sync-To' => self.sync_to}) unless self.sync_to.nil?
|
|
options.merge!({'X-Container-Sync-Key' => self.sync_key}) unless self.sync_key.nil?
|
|
options.merge!({'X-Container-Meta-Web-Index' => self.web_index}) unless self.web_index.nil?
|
|
options.merge!({'X-Container-Meta-Web-Listings' => self.web_listings})
|
|
options.merge!({'X-Container-Meta-Web-Listings-Css' => self.web_listings_css}) unless self.web_listings_css.nil?
|
|
options.merge!({'X-Container-Meta-Web-Error' => self.web_error}) unless self.web_error.nil?
|
|
# get the metadata and merge them in
|
|
|
|
# merge user options at the end
|
|
options.merge!(new_options)
|
|
|
|
service.put_container(key, options)
|
|
# Added an extra check to see if CDN is enabled for the container
|
|
if (!service.cdn.nil? && service.cdn.enabled?)
|
|
# If CDN available, set the container to be CDN-enabled or not based on if it is marked as cdn_enable.
|
|
if @cdn_enable
|
|
# check to make sure that the container exists. If yes, cdn enable it.
|
|
begin response = service.cdn.head_container(key)
|
|
if response.headers['X-Cdn-Enabled'] == 'False'
|
|
service.cdn.post_container(key, {'X-Cdn-Enabled' => 'True'})
|
|
end
|
|
rescue Fog::CDN::HP::NotFound
|
|
service.cdn.put_container(key)
|
|
rescue Excon::Errors::SocketError
|
|
# means that the CDN endpoint is unreachable
|
|
end
|
|
else
|
|
# check to make sure that the container exists. If yes, cdn disable it.
|
|
begin response = service.cdn.head_container(key)
|
|
### Deleting a container from CDN is much more expensive than flipping the bit to disable it
|
|
if response.headers['X-Cdn-Enabled'] == 'True'
|
|
service.cdn.post_container(key, {'X-Cdn-Enabled' => 'False'})
|
|
end
|
|
rescue Fog::CDN::HP::NotFound
|
|
# just continue, as container is not cdn-enabled.
|
|
rescue Excon::Errors::SocketError
|
|
# means that the CDN endpoint is unreachable
|
|
end
|
|
end
|
|
end
|
|
true
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|