file.rb 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. # Copyright (C) 2012-2016 Zammad Foundation, http://zammad-foundation.org/
  2. class Store
  3. class File < ApplicationModel
  4. include ApplicationLib
  5. after_destroy :destroy_provider
  6. =begin
  7. add new file to store
  8. store_file_id = Store::File.add(binary_data)
  9. do also verify of written data
  10. store_file_id = Store::File.add(binary_data, true)
  11. =end
  12. def self.add(data, verify = true)
  13. sha = Digest::SHA256.hexdigest(data)
  14. file = Store::File.find_by(sha: sha)
  15. if file.nil?
  16. # load backend based on config
  17. adapter_name = Setting.get('storage_provider') || 'DB'
  18. if !adapter_name
  19. raise 'Missing storage_provider setting option'
  20. end
  21. adapter = load_adapter("Store::Provider::#{adapter_name}")
  22. adapter.add(data, sha)
  23. file = Store::File.create(
  24. provider: adapter_name,
  25. sha: sha,
  26. )
  27. # verify
  28. if verify
  29. read_data = adapter.get(sha)
  30. read_sha = Digest::SHA256.hexdigest(read_data)
  31. if sha != read_sha
  32. raise "Content not written correctly (provider #{adapter_name})."
  33. end
  34. end
  35. end
  36. file
  37. end
  38. =begin
  39. read content of a file
  40. store = Store::File.find(123)
  41. store.content # returns binary
  42. =end
  43. def content
  44. adapter = self.class.load_adapter("Store::Provider::#{provider}")
  45. adapter.get(sha)
  46. end
  47. =begin
  48. file system check of store, check data and sha (in case fix it)
  49. Store::File.verify
  50. read each file which should be in backend and verify agsinst sha hash
  51. in case of fixing sha hash use:
  52. Store::File.verify(true)
  53. =end
  54. def self.verify(fix_it = nil)
  55. success = true
  56. file_ids = Store::File.all.pluck(:id)
  57. file_ids.each { |item_id|
  58. item = Store::File.find(item_id)
  59. content = item.content
  60. sha = Digest::SHA256.hexdigest(content)
  61. logger.info "CHECK: Store::File.find(#{item.id})"
  62. next if sha == item.sha
  63. success = false
  64. logger.error "DIFF: sha diff of Store::File.find(#{item.id}) current:#{sha}/db:#{item.sha}/provider:#{item.provider}"
  65. store = Store.find_by(store_file_id: item.id)
  66. logger.error "STORE: #{store.inspect}"
  67. if fix_it
  68. item.update_attribute(:sha, sha)
  69. end
  70. }
  71. success
  72. end
  73. =begin
  74. move file from one to other provider
  75. move files from file backend to db
  76. Store::File.move('File', 'DB')
  77. move files from db backend to fs
  78. Store::File.move('DB', 'File')
  79. nice move to keep system responsive
  80. Store::File.move('DB', 'File', delay_in_sec) # e. g. 1
  81. =end
  82. def self.move(source, target, delay = nil)
  83. adapter_source = load_adapter("Store::Provider::#{source}")
  84. adapter_target = load_adapter("Store::Provider::#{target}")
  85. file_ids = Store::File.all.pluck(:id)
  86. file_ids.each { |item_id|
  87. item = Store::File.find(item_id)
  88. next if item.provider == target
  89. content = item.content
  90. # add to new provider
  91. adapter_target.add(content, item.sha)
  92. # update meta data
  93. item.update_attribute(:provider, target)
  94. # remove from old provider
  95. adapter_source.delete(item.sha)
  96. logger.info "Moved file #{item.sha} from #{source} to #{target}"
  97. sleep delay if delay
  98. }
  99. true
  100. end
  101. private
  102. def destroy_provider
  103. adapter = self.class.load_adapter("Store::Provider::#{provider}")
  104. adapter.delete(sha)
  105. end
  106. end
  107. end