From ce5dfcfa6a67b7683ac21f68867e9483c11f4fbe Mon Sep 17 00:00:00 2001 From: Ari Timonen Date: Wed, 1 Apr 2020 05:00:53 +0300 Subject: [PATCH] Finish recreate, Fix file browsing Add it to adminpanel --- .../themes/default/layout/_body.scss | 4 + app/controllers/data_files_controller.rb | 2 +- app/controllers/directories_controller.rb | 12 +- app/helpers/application_helper.rb | 19 ++- app/models/data_file.rb | 16 +- app/models/directory.rb | 145 ++++++++++++------ app/uploaders/file_uploader.rb | 3 +- app/views/about/adminpanel.html.erb | 44 +++++- app/views/data_files/edit.html.erb | 2 +- app/views/directories/_form.html.erb | 2 +- app/views/directories/recreate.html.erb | 5 + app/views/directories/show.html.erb | 46 +++--- config/routes.rb | 6 +- 13 files changed, 212 insertions(+), 94 deletions(-) create mode 100644 app/views/directories/recreate.html.erb diff --git a/app/assets/stylesheets/themes/default/layout/_body.scss b/app/assets/stylesheets/themes/default/layout/_body.scss index 75299cc..aeb930a 100644 --- a/app/assets/stylesheets/themes/default/layout/_body.scss +++ b/app/assets/stylesheets/themes/default/layout/_body.scss @@ -131,4 +131,8 @@ h1, h2, h3, h4, h5, h6 { ol { list-style-type: decimal; } + + pre { + font-size: $base-font-size * 0.5; + } } diff --git a/app/controllers/data_files_controller.rb b/app/controllers/data_files_controller.rb index 7faa1f1..105fdf2 100644 --- a/app/controllers/data_files_controller.rb +++ b/app/controllers/data_files_controller.rb @@ -74,7 +74,7 @@ class DataFilesController < ApplicationController def destroy raise AccessError unless @file.can_destroy? cuser @file.destroy - redirect_to_back + redirect_to directory_path(@file.directory) end def rate diff --git a/app/controllers/directories_controller.rb b/app/controllers/directories_controller.rb index 21dc1af..8c206eb 100644 --- a/app/controllers/directories_controller.rb +++ b/app/controllers/directories_controller.rb @@ -21,13 +21,9 @@ class DirectoriesController < ApplicationController end def recreate - @directory.recreate_transaction - render text: t(:directories_update) - end - - def refresh - @directory.process_dir - render text: t(:directories_update) + raise AccessError unless @cuser&.admin? + @result = @directory.recreate_transaction + @nobody = true end def create @@ -55,7 +51,7 @@ class DirectoriesController < ApplicationController def destroy raise AccessError unless @directory.can_destroy? cuser @directory.destroy - redirect_to directories_url + redirect_to directory_path(Directory.find(Directory::ROOT)) end private diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb index 793a66b..797c494 100644 --- a/app/helpers/application_helper.rb +++ b/app/helpers/application_helper.rb @@ -22,7 +22,7 @@ module ApplicationHelper stylesheet_link_tag "themes/#{active_theme}/theme" end - def namelink model, length = nil + def namelink(model, length = nil) return if model.nil? model = case model.class.to_s when "DataFile" @@ -35,15 +35,26 @@ module ApplicationHelper model end str = model.to_s - + # Reduce length of too long model names if length and str.length > length - link_to str.to_s[0, length] + "...", model, class: model.class.to_s.downcase + link_to(str.to_s[0, length] + "...", model, class: model.class.to_s.downcase) else - link_to str, model, class: model.class.to_s.downcase + link_to(str, model, class: model.class.to_s.downcase) end end + def directory_links(directory) + output = "" + Directory.directory_traverse(directory).reverse_each do |dir| + output << namelink(dir) + "\n" + unless dir == directory + output << " » \n" + end + end + output.html_safe + end + def shorten str, length if length and str and str.to_s.length > length str = str.to_s[0, length] + "..." diff --git a/app/models/data_file.rb b/app/models/data_file.rb index 72a05ec..7eac3b1 100644 --- a/app/models/data_file.rb +++ b/app/models/data_file.rb @@ -83,6 +83,12 @@ class DataFile < ActiveRecord::Base name.url end + def manual_upload(manual_location) + File.open(manual_location) do |f| + self.name = f + end + end + def process_file self.md5 = "e948c22100d29623a1df48e1760494df" @@ -90,7 +96,7 @@ class DataFile < ActiveRecord::Base self.directory_id = Directory::ARTICLES end - if File.exists?(location) and (size != File.size(location) or created_at != File.mtime(location)) + if File.exists?(location) and (new_record? or size != File.size(location) or created_at != File.mtime(location)) self.md5 = Digest::MD5.hexdigest(File.read(location)) self.size = File.size(location) self.created_at = File.mtime(location) @@ -147,9 +153,11 @@ class DataFile < ActiveRecord::Base user and !rated_by?(user) end - def self.find_existing(subdir_name, subitem_path) - DataFile.where(arel_tabe(:path).eq(subitem_path)\ - .or(arel_table(:md5).eq(Digest::MD5.hexdigest(File.read(subitem_path))))).first + # TODO: instead of using path, use name + directory path + def self.find_existing(subitem_path, subitem_name) + hash = Digest::MD5.hexdigest(File.read(subitem_path)) + DataFile.where(arel_table[:path].eq(subitem_path)\ + .or(arel_table[:md5].eq(hash))).first end def can_create? cuser diff --git a/app/models/directory.rb b/app/models/directory.rb index 4723a32..a4b351a 100644 --- a/app/models/directory.rb +++ b/app/models/directory.rb @@ -17,19 +17,16 @@ # # index_directories_on_parent_id (parent_id) # +require 'stringio' ENV['FILES_ROOT'] ||= File.join(Rails.root, 'public', 'files') -# class SteamIdValidator < ActiveModel::Validator -# def validate(record) -# record.errors.add :steamid unless \ -# record.steamid.nil? || -# (m = record.steamid.match(/\A([01]):([01]):(\d{1,10})\Z/)) && -# (id = m[3].to_i) && -# id >= 1 && id <= 2147483647 -# end -# end - +class PathValidator < ActiveModel::Validator + def validate(record) + record.errors.add :path, "doesn't match generated path" unless \ + record.full_path == record.path + end +end class Directory < ActiveRecord::Base include Extra @@ -41,43 +38,84 @@ class Directory < ActiveRecord::Base MOVIES = 30 ARTICLES = 39 - #attr_protected :id, :updated_at, :created_at, :path + attr_accessor :preserve_files belongs_to :parent, :class_name => "Directory", :optional => true has_many :subdirs, :class_name => "Directory", :foreign_key => :parent_id has_many :files, -> { order("name") }, :class_name => "DataFile" scope :ordered, -> { order("name ASC") } + scope :path_sorted, -> { order("path ASC") } scope :filtered, -> { where(hidden: false) } scope :of_parent, -> (parent) { where(parent_id: parent.id) } - validates_length_of [:name, :path], :in => 1..255 + # FIXME: different validation for user? + validates_length_of [:name, :path, :title], :in => 1..255 validates_format_of :name, :with => /\A[A-Za-z0-9]{1,20}\z/, :on => :create - validates_length_of :name, :in => 1..25 + validates_length_of :name, :in => 1..255 validates_inclusion_of :hidden, :in => [true, false] + validates_presence_of :title + validates_with PathValidator # TODO: add validation for path - before_validation :init_variables + before_validation :init_variables, on: :create after_create :make_path after_save :update_timestamp - before_destroy :remove_files + before_destroy :remove_files, unless: Proc.new { preserve_files } after_destroy :remove_path def to_s name end - def init_variables - self.path = full_path if parent - self.hidden = false if hidden.nil? + def parent_root? + parent.id == Directory::ROOT + end + + def root? + id == Directory::ROOT + end + + def full_title + output = "" + Directory.directory_traverse(self).reverse_each do |dir| + unless dir.title&.empty? + output << "%s" % dir.title + else + output << dir.name + end + output << " ยป " unless self == dir + end + output + end + + def self.directory_traverse(directory, list = []) + unless directory.root? + list << directory + return directory_traverse(directory.parent, list) + else + return list + end + end + + # Use this + def full_path + parent ? File.join(parent.full_path, name.downcase) : path + end + + def relative_path + parent ? File.join(parent.relative_path, name.downcase).sub(/^\//, '') : "" end def path_exists? File.directory?(full_path) end - def full_path - parent ? File.join(parent.full_path, name.downcase) : path + def init_variables + # Force path to use parent which is the authoritative source + self.path = full_path if parent + self.title = File.basename(self.path).capitalize + self.hidden = false if hidden.nil? end def make_path @@ -93,6 +131,7 @@ class Directory < ActiveRecord::Base subdir.destroy end subdirs.each do |subdir| + subdir.preserve_files = self.preserve_files subdir.destroy end end @@ -103,65 +142,86 @@ class Directory < ActiveRecord::Base # TODO: make tests for this, moving etc. # TODO: mutate instead of return. - def recreate_transaction(root = ENV['FILES_ROOT']) - logger = Rails.logger - logger.info 'Starting recreate on %d, root: %s' % [id, root] + # TODO: move to its own class + # TODO: also remove files + # TODO: need log to rails log too + def recreate_transaction + strio = StringIO.new + logger = Logger.new(strio) + logger.info 'Starting recreate on Directory(%d): %s.' % [id, name] ActiveRecord::Base.transaction do # We use destroy lists so technically there can be seperate roots destroy_dirs = Hash.new - update_attribute :path, root - destroy_dirs = recreate(destroy_dirs) + if id == Directory::ROOT + update_attribute :path, ENV['FILES_ROOT'] + end + logger.info 'Path: %s' % [path] + destroy_dirs = recreate(destroy_dirs, logger: logger) destroy_dirs.each do |key, dir| logger.info 'Removed dir: %s' % dir.full_path - # dir.destroy! + dir.destroy! end end logger.info 'Finish recreate' - return nil + return strio # TODO: check items that weren't checked. end # QUESTION Symlinks? - def recreate(destroy_dirs, path = self.full_path) + def recreate(destroy_dirs, logger: Rails.logger) # Convert all subdirs into a hash and mark them to be deleted # FIXME: better oneliner + # logger.debug 'recreate: %s' % full_path destroy_dirs.merge!(subdirs.all.map{ |s| [s.id,s] }.to_h) # Go through all subdirectories (no recursion) - Dir.glob("%s/*" % path).each do |subitem_path| - if File.directory? subitem_path - subdir_name = File.basename(subitem_path) + Dir.glob(File.join(full_path, '*')).each do |subitem_path| + subitem_name = File.basename(subitem_path) + if File.directory? subitem_path + # logger.debug 'Processing dir: %s' % subitem_path # We find by name only, ignore path # Find existing subdirs from current path. Keep those we find - if (subdir = find_existing(subdir_name, subitem_path)) + if (subdir = find_existing(subitem_name, subitem_path)) if subdir.parent_id != self.id old_path = subdir.full_path subdir.parent = self subdir.save! logger.info 'Renamed dir: %s -> %s' % [old_path, subdir.full_path] + elsif !subdir.valid? + subdir.errors.full_messages.each do |err| + logger.error err + end + subdir.init_variables + logger.info 'Fixed attributes: %s' % [subdir.full_path] + subdir.save! end destroy_dirs.delete subdir.id # In case its a new directory else # Attempt to find it in existing directories - subdir = subdirs.build(name: subdir_name) + subdir = subdirs.build(name: subitem_name) # FIXME: find a better solution subdir.save!(validate: false) logger.info 'New dir: %s' % subdir.full_path end - # Recreate the directory - destroy_dirs = subdir.recreate(destroy_dirs) + destroy_dirs = subdir.recreate(destroy_dirs, logger: logger) elsif File.file? subitem_path + # logger.debug 'Processing file: %s' % subitem_path if dbfile = DataFile.find_existing(subitem_path, subitem_name) - dbfile.directory = self - dbfile.save! - elsif (File.mtime(file) + 100).past? + if dbfile.directory_id != self.id + dbfile.directory = self + dbfile.save! + logger.info 'Update file: %s' % dbfile.name + end + elsif (File.mtime(subitem_path) + 100).past? dbfile = DataFile.new - dbfile.path = file + # dbfile.name = subitem_name dbfile.directory = self + dbfile.manual_upload(subitem_path) dbfile.save! + logger.info 'Added file: %s' % dbfile.name end # TODO: handle files that are only in database end @@ -181,7 +241,7 @@ class Directory < ActiveRecord::Base end end # TODO: use filter_map here - # NOTE: we don't use the logic from dat_file + # NOTE: we don't use the logic from date_file file_count = Dir["%s/*" % subitem_path].count{|f| File.file?(f) } Directory.joins(:files).group('data_files.directory_id')\ .having('count(data_files.id) = ? and count(data_files.id) > 0', file_count).each do |dir| @@ -195,11 +255,6 @@ class Directory < ActiveRecord::Base return false end - # TODO - def recreate_check - true - end - # TODO check that you can download files def can_create? cuser diff --git a/app/uploaders/file_uploader.rb b/app/uploaders/file_uploader.rb index 8d8a8c0..5b9005c 100644 --- a/app/uploaders/file_uploader.rb +++ b/app/uploaders/file_uploader.rb @@ -4,7 +4,8 @@ class FileUploader < CarrierWave::Uploader::Base # Override the directory where uploaded files will be stored. # This is a sensible default for uploaders that are meant to be mounted: def store_dir - model.directory.path.gsub(/public\//, '') + model.directory.full_path + # .gsub(/public\//, '') end # Provide a default URL as a default if there hasn't been a file uploaded: diff --git a/app/views/about/adminpanel.html.erb b/app/views/about/adminpanel.html.erb index 997a4a5..595da52 100644 --- a/app/views/about/adminpanel.html.erb +++ b/app/views/about/adminpanel.html.erb @@ -2,25 +2,53 @@

Admin Menu

+

+ Articles +

+ +

+ Files +

+ + +

+ Contests +

+ + +

+ Users and groups +

+ + +

+ Website +

+ \ No newline at end of file diff --git a/app/views/data_files/edit.html.erb b/app/views/data_files/edit.html.erb index ba510c9..dae2693 100644 --- a/app/views/data_files/edit.html.erb +++ b/app/views/data_files/edit.html.erb @@ -12,7 +12,7 @@
<%= f.label :directory_id %> - <%= f.select :directory_id, Directory.all.collect { |c| ["#{c.path} - #{c.name}", c.id] } %> + <%= f.select :directory_id, Directory.path_sorted.collect { |c| ["#{c.full_title} (#{c.relative_path})", c.id] } %>
diff --git a/app/views/directories/_form.html.erb b/app/views/directories/_form.html.erb index a7a35e5..0e17a86 100644 --- a/app/views/directories/_form.html.erb +++ b/app/views/directories/_form.html.erb @@ -12,7 +12,7 @@ <% end %>

- Parent: <%= @directory.parent.path %> + Parent: <%= @directory.parent.full_path %>

diff --git a/app/views/directories/recreate.html.erb b/app/views/directories/recreate.html.erb new file mode 100644 index 0000000..91598ad --- /dev/null +++ b/app/views/directories/recreate.html.erb @@ -0,0 +1,5 @@ +<%= t(:directories_update) %> + +

+  <%= @result.string %>
+
\ No newline at end of file diff --git a/app/views/directories/show.html.erb b/app/views/directories/show.html.erb index 6c6b969..7882312 100644 --- a/app/views/directories/show.html.erb +++ b/app/views/directories/show.html.erb @@ -1,6 +1,14 @@

Files

-<% +

+<% if @directory.root? %> + In case there are any problems with this browser, you can find files + <%= link_to "here", "/files/" %> + too. +<% end%> +

+ +<% active = 1 n = 1 %> @@ -15,9 +23,8 @@
<% @directories.each do |dir| %>
- <% - if @directory.path.include?(dir.path) + if @directory.full_path.include?(dir.full_path) dir = @directory active = n end @@ -25,24 +32,23 @@ %>
- <% if !dir.subdirs.ordered.empty? or dir.parent.id != Directory::ROOT %> -

Directories

+ <% if !dir.parent_root? %> +

<%= directory_links @directory %>

+ <% end %> + <% if !dir.subdirs.ordered.empty? %> + <% if dir.parent_root? %> +

Sub-directories

+ <% end %> +
+
    + <% dir.subdirs.ordered.each do |subdir| %> +
  • <%= link_to subdir.name, subdir %>
  • + <% end %> +
+
<% end %> - -
-
    - <% if dir.parent.id != Directory::ROOT %> -
  • - <%= link_to "Parent", dir.parent %> -
  • - <% end %> - <% dir.subdirs.ordered.each do |subdir| %> -
  • <%= namelink subdir %>
  • - <% end %> -
-
- +

Files

<% dir.files.unrelated.each do |file| %> @@ -77,7 +83,7 @@ <% if cuser and cuser.admin? %>
<%= link_to "Edit Directory", edit_directory_path(dir), class: 'button' %> - <%= link_to "Delete Directory", dir, confirm: "Are you REALLY sure?", method: :delete, class: 'button' %> + <%= link_to "Delete Directory", dir, data: { confirm: "Are you sure?" }, class: 'button' %> <%= link_to "New Directory", { controller: "directories", action: "new", id: dir }, { class: 'button' } %> <%= link_to "New File", { controller: "data_files", action: "new", id: dir }, { class: 'button' } %>
diff --git a/config/routes.rb b/config/routes.rb index e96f736..06bce47 100755 --- a/config/routes.rb +++ b/config/routes.rb @@ -85,7 +85,11 @@ Ensl::Application.routes.draw do resources :maps resources :logs resources :log_files - resources :directories + resources :directories do + member do + get :recreate + end + end resources :data_files resources :predictions resources :weeks