Finish recreate, Fix file browsing

Add it to adminpanel
This commit is contained in:
Ari Timonen 2020-04-01 05:00:53 +03:00
parent b571d6e2c2
commit ce5dfcfa6a
13 changed files with 212 additions and 94 deletions

View file

@ -131,4 +131,8 @@ h1, h2, h3, h4, h5, h6 {
ol { ol {
list-style-type: decimal; list-style-type: decimal;
} }
pre {
font-size: $base-font-size * 0.5;
}
} }

View file

@ -74,7 +74,7 @@ class DataFilesController < ApplicationController
def destroy def destroy
raise AccessError unless @file.can_destroy? cuser raise AccessError unless @file.can_destroy? cuser
@file.destroy @file.destroy
redirect_to_back redirect_to directory_path(@file.directory)
end end
def rate def rate

View file

@ -21,13 +21,9 @@ class DirectoriesController < ApplicationController
end end
def recreate def recreate
@directory.recreate_transaction raise AccessError unless @cuser&.admin?
render text: t(:directories_update) @result = @directory.recreate_transaction
end @nobody = true
def refresh
@directory.process_dir
render text: t(:directories_update)
end end
def create def create
@ -55,7 +51,7 @@ class DirectoriesController < ApplicationController
def destroy def destroy
raise AccessError unless @directory.can_destroy? cuser raise AccessError unless @directory.can_destroy? cuser
@directory.destroy @directory.destroy
redirect_to directories_url redirect_to directory_path(Directory.find(Directory::ROOT))
end end
private private

View file

@ -22,7 +22,7 @@ module ApplicationHelper
stylesheet_link_tag "themes/#{active_theme}/theme" stylesheet_link_tag "themes/#{active_theme}/theme"
end end
def namelink model, length = nil def namelink(model, length = nil)
return if model.nil? return if model.nil?
model = case model.class.to_s model = case model.class.to_s
when "DataFile" when "DataFile"
@ -35,15 +35,26 @@ module ApplicationHelper
model model
end end
str = model.to_s str = model.to_s
# Reduce length of too long model names # Reduce length of too long model names
if length and str.length > length 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 else
link_to str, model, class: model.class.to_s.downcase link_to(str, model, class: model.class.to_s.downcase)
end end
end end
def directory_links(directory)
output = ""
Directory.directory_traverse(directory).reverse_each do |dir|
output << namelink(dir) + "\n"
unless dir == directory
output << " &raquo; \n"
end
end
output.html_safe
end
def shorten str, length def shorten str, length
if length and str and str.to_s.length > length if length and str and str.to_s.length > length
str = str.to_s[0, length] + "..." str = str.to_s[0, length] + "..."

View file

@ -83,6 +83,12 @@ class DataFile < ActiveRecord::Base
name.url name.url
end end
def manual_upload(manual_location)
File.open(manual_location) do |f|
self.name = f
end
end
def process_file def process_file
self.md5 = "e948c22100d29623a1df48e1760494df" self.md5 = "e948c22100d29623a1df48e1760494df"
@ -90,7 +96,7 @@ class DataFile < ActiveRecord::Base
self.directory_id = Directory::ARTICLES self.directory_id = Directory::ARTICLES
end 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.md5 = Digest::MD5.hexdigest(File.read(location))
self.size = File.size(location) self.size = File.size(location)
self.created_at = File.mtime(location) self.created_at = File.mtime(location)
@ -147,9 +153,11 @@ class DataFile < ActiveRecord::Base
user and !rated_by?(user) user and !rated_by?(user)
end end
def self.find_existing(subdir_name, subitem_path) # TODO: instead of using path, use name + directory path
DataFile.where(arel_tabe(:path).eq(subitem_path)\ def self.find_existing(subitem_path, subitem_name)
.or(arel_table(:md5).eq(Digest::MD5.hexdigest(File.read(subitem_path))))).first hash = Digest::MD5.hexdigest(File.read(subitem_path))
DataFile.where(arel_table[:path].eq(subitem_path)\
.or(arel_table[:md5].eq(hash))).first
end end
def can_create? cuser def can_create? cuser

View file

@ -17,19 +17,16 @@
# #
# index_directories_on_parent_id (parent_id) # index_directories_on_parent_id (parent_id)
# #
require 'stringio'
ENV['FILES_ROOT'] ||= File.join(Rails.root, 'public', 'files') ENV['FILES_ROOT'] ||= File.join(Rails.root, 'public', 'files')
# class SteamIdValidator < ActiveModel::Validator class PathValidator < ActiveModel::Validator
# def validate(record) def validate(record)
# record.errors.add :steamid unless \ record.errors.add :path, "doesn't match generated path" unless \
# record.steamid.nil? || record.full_path == record.path
# (m = record.steamid.match(/\A([01]):([01]):(\d{1,10})\Z/)) && end
# (id = m[3].to_i) && end
# id >= 1 && id <= 2147483647
# end
# end
class Directory < ActiveRecord::Base class Directory < ActiveRecord::Base
include Extra include Extra
@ -41,43 +38,84 @@ class Directory < ActiveRecord::Base
MOVIES = 30 MOVIES = 30
ARTICLES = 39 ARTICLES = 39
#attr_protected :id, :updated_at, :created_at, :path attr_accessor :preserve_files
belongs_to :parent, :class_name => "Directory", :optional => true belongs_to :parent, :class_name => "Directory", :optional => true
has_many :subdirs, :class_name => "Directory", :foreign_key => :parent_id has_many :subdirs, :class_name => "Directory", :foreign_key => :parent_id
has_many :files, -> { order("name") }, :class_name => "DataFile" has_many :files, -> { order("name") }, :class_name => "DataFile"
scope :ordered, -> { order("name ASC") } scope :ordered, -> { order("name ASC") }
scope :path_sorted, -> { order("path ASC") }
scope :filtered, -> { where(hidden: false) } scope :filtered, -> { where(hidden: false) }
scope :of_parent, -> (parent) { where(parent_id: parent.id) } 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_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_inclusion_of :hidden, :in => [true, false]
validates_presence_of :title
validates_with PathValidator
# TODO: add validation for path # TODO: add validation for path
before_validation :init_variables before_validation :init_variables, on: :create
after_create :make_path after_create :make_path
after_save :update_timestamp after_save :update_timestamp
before_destroy :remove_files before_destroy :remove_files, unless: Proc.new { preserve_files }
after_destroy :remove_path after_destroy :remove_path
def to_s def to_s
name name
end end
def init_variables def parent_root?
self.path = full_path if parent parent.id == Directory::ROOT
self.hidden = false if hidden.nil? 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 end
def path_exists? def path_exists?
File.directory?(full_path) File.directory?(full_path)
end end
def full_path def init_variables
parent ? File.join(parent.full_path, name.downcase) : path # 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 end
def make_path def make_path
@ -93,6 +131,7 @@ class Directory < ActiveRecord::Base
subdir.destroy subdir.destroy
end end
subdirs.each do |subdir| subdirs.each do |subdir|
subdir.preserve_files = self.preserve_files
subdir.destroy subdir.destroy
end end
end end
@ -103,65 +142,86 @@ class Directory < ActiveRecord::Base
# TODO: make tests for this, moving etc. # TODO: make tests for this, moving etc.
# TODO: mutate instead of return. # TODO: mutate instead of return.
def recreate_transaction(root = ENV['FILES_ROOT']) # TODO: move to its own class
logger = Rails.logger # TODO: also remove files
logger.info 'Starting recreate on %d, root: %s' % [id, root] # 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 ActiveRecord::Base.transaction do
# We use destroy lists so technically there can be seperate roots # We use destroy lists so technically there can be seperate roots
destroy_dirs = Hash.new destroy_dirs = Hash.new
update_attribute :path, root if id == Directory::ROOT
destroy_dirs = recreate(destroy_dirs) 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| destroy_dirs.each do |key, dir|
logger.info 'Removed dir: %s' % dir.full_path logger.info 'Removed dir: %s' % dir.full_path
# dir.destroy! dir.destroy!
end end
end end
logger.info 'Finish recreate' logger.info 'Finish recreate'
return nil return strio
# TODO: check items that weren't checked. # TODO: check items that weren't checked.
end end
# QUESTION Symlinks? # 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 # Convert all subdirs into a hash and mark them to be deleted
# FIXME: better oneliner # FIXME: better oneliner
# logger.debug 'recreate: %s' % full_path
destroy_dirs.merge!(subdirs.all.map{ |s| [s.id,s] }.to_h) destroy_dirs.merge!(subdirs.all.map{ |s| [s.id,s] }.to_h)
# Go through all subdirectories (no recursion) # Go through all subdirectories (no recursion)
Dir.glob("%s/*" % path).each do |subitem_path| Dir.glob(File.join(full_path, '*')).each do |subitem_path|
if File.directory? subitem_path subitem_name = File.basename(subitem_path)
subdir_name = File.basename(subitem_path)
if File.directory? subitem_path
# logger.debug 'Processing dir: %s' % subitem_path
# We find by name only, ignore path # We find by name only, ignore path
# Find existing subdirs from current path. Keep those we find # 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 if subdir.parent_id != self.id
old_path = subdir.full_path old_path = subdir.full_path
subdir.parent = self subdir.parent = self
subdir.save! subdir.save!
logger.info 'Renamed dir: %s -> %s' % [old_path, subdir.full_path] 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 end
destroy_dirs.delete subdir.id destroy_dirs.delete subdir.id
# In case its a new directory # In case its a new directory
else else
# Attempt to find it in existing directories # Attempt to find it in existing directories
subdir = subdirs.build(name: subdir_name) subdir = subdirs.build(name: subitem_name)
# FIXME: find a better solution # FIXME: find a better solution
subdir.save!(validate: false) subdir.save!(validate: false)
logger.info 'New dir: %s' % subdir.full_path logger.info 'New dir: %s' % subdir.full_path
end end
# Recreate the directory # Recreate the directory
destroy_dirs = subdir.recreate(destroy_dirs) destroy_dirs = subdir.recreate(destroy_dirs, logger: logger)
elsif File.file? subitem_path elsif File.file? subitem_path
# logger.debug 'Processing file: %s' % subitem_path
if dbfile = DataFile.find_existing(subitem_path, subitem_name) if dbfile = DataFile.find_existing(subitem_path, subitem_name)
dbfile.directory = self if dbfile.directory_id != self.id
dbfile.save! dbfile.directory = self
elsif (File.mtime(file) + 100).past? dbfile.save!
logger.info 'Update file: %s' % dbfile.name
end
elsif (File.mtime(subitem_path) + 100).past?
dbfile = DataFile.new dbfile = DataFile.new
dbfile.path = file # dbfile.name = subitem_name
dbfile.directory = self dbfile.directory = self
dbfile.manual_upload(subitem_path)
dbfile.save! dbfile.save!
logger.info 'Added file: %s' % dbfile.name
end end
# TODO: handle files that are only in database # TODO: handle files that are only in database
end end
@ -181,7 +241,7 @@ class Directory < ActiveRecord::Base
end end
end end
# TODO: use filter_map here # 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) } file_count = Dir["%s/*" % subitem_path].count{|f| File.file?(f) }
Directory.joins(:files).group('data_files.directory_id')\ Directory.joins(:files).group('data_files.directory_id')\
.having('count(data_files.id) = ? and count(data_files.id) > 0', file_count).each do |dir| .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 return false
end end
# TODO
def recreate_check
true
end
# TODO check that you can download files # TODO check that you can download files
def can_create? cuser def can_create? cuser

View file

@ -4,7 +4,8 @@ class FileUploader < CarrierWave::Uploader::Base
# Override the directory where uploaded files will be stored. # Override the directory where uploaded files will be stored.
# This is a sensible default for uploaders that are meant to be mounted: # This is a sensible default for uploaders that are meant to be mounted:
def store_dir def store_dir
model.directory.path.gsub(/public\//, '') model.directory.full_path
# .gsub(/public\//, '')
end end
# Provide a default URL as a default if there hasn't been a file uploaded: # Provide a default URL as a default if there hasn't been a file uploaded:

View file

@ -2,25 +2,53 @@
<h1> <h1>
Admin Menu Admin Menu
</h1> </h1>
<h4>
Articles
</h4>
<ul class="disc"> <ul class="disc">
<li><%= link_to 'New Article', new_article_path %></li> <li><%= link_to 'New Article', new_article_path %></li>
<li><%= link_to "Article Admin", controller: "articles", action: "admin" %></li> <li><%= link_to "Article Admin", controller: "articles", action: "admin" %></li>
</ul>
<h4>
Files
</h4>
<ul class="disc">
<% if Directory.first %> <% if Directory.first %>
<li><%= link_to "Recreate Root", recreate_directory_path(Directory.find(Directory::ROOT)) %></li>
<li><%= link_to "Files Admin", controller: "directories", action: "show", id: Directory.first %></li> <li><%= link_to "Files Admin", controller: "directories", action: "show", id: Directory.first %></li>
<% end %> <% end %>
</ul>
<h4>
Contests
</h4>
<ul class="disc">
<li><%= link_to "Contests", contests_path %></li>
<li><%= link_to "Challenges", challenges_path %></li>
<li><%= link_to "Maps", maps_path %></li>
</ul>
<h4>
Users and groups
</h4>
<ul class="disc">
<li><%= link_to "Users", users_path %></li>
<li><%= link_to "Groups", groups_path %></li>
<li><%= link_to "Bans", bans_path %></li>
</ul>
<h4>
Website
</h4>
<ul class="disc">
<li><%= link_to "Categories", categories_path %></li>
<li><%= link_to "Custom Article URLs", custom_urls_path %></li>
<li> <li>
<%= link_to issues_path do %> <%= link_to issues_path do %>
Issues (<%= Issue.with_status(0).count %>) Issues (<%= Issue.with_status(0).count %>)
<% end %> <% end %>
</li> </li>
<li><%= link_to "Bans", bans_path %></li>
<li><%= link_to "Groups", groups_path %></li>
<li><%= link_to "Categories", categories_path %></li>
<li><%= link_to "Polls", polls_path %></li> <li><%= link_to "Polls", polls_path %></li>
<li><%= link_to "Contests", contests_path %></li>
<li><%= link_to "Challenges", challenges_path %></li>
<li><%= link_to "Maps", maps_path %></li>
<li><%= link_to "Users", users_path %></li>
<li><%= link_to "Custom Article URLs", custom_urls_path %></li>
</ul> </ul>
</div> </div>

View file

@ -12,7 +12,7 @@
<div class="fields horizontal"> <div class="fields horizontal">
<%= f.label :directory_id %> <%= 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] } %>
</div> </div>
<div class="controls"> <div class="controls">

View file

@ -12,7 +12,7 @@
<% end %> <% end %>
<p> <p>
<b>Parent:</b> <%= @directory.parent.path %> <b>Parent:</b> <%= @directory.parent.full_path %>
</p> </p>
<p> <p>

View file

@ -0,0 +1,5 @@
<%= t(:directories_update) %>
<pre>
<%= @result.string %>
</pre>

View file

@ -1,6 +1,14 @@
<h1 class="title">Files</h1> <h1 class="title">Files</h1>
<% <p>
<% if @directory.root? %>
In case there are any problems with this browser, you can find files
<%= link_to "here", "/files/" %>
too.
<% end%>
</p>
<%
active = 1 active = 1
n = 1 n = 1
%> %>
@ -15,9 +23,8 @@
<div class="tabbed-contents"> <div class="tabbed-contents">
<% @directories.each do |dir| %> <% @directories.each do |dir| %>
<div class="tab" id="dir_<%= dir.id %>"> <div class="tab" id="dir_<%= dir.id %>">
<% <%
if @directory.path.include?(dir.path) if @directory.full_path.include?(dir.full_path)
dir = @directory dir = @directory
active = n active = n
end end
@ -25,24 +32,23 @@
%> %>
<div class="directories"> <div class="directories">
<% if !dir.subdirs.ordered.empty? or dir.parent.id != Directory::ROOT %> <% if !dir.parent_root? %>
<h3>Directories</h3> <h3><%= directory_links @directory %></h3>
<% end %>
<% if !dir.subdirs.ordered.empty? %>
<% if dir.parent_root? %>
<h3>Sub-directories</h3>
<% end %>
<div class="subdirectories">
<ul class="disc">
<% dir.subdirs.ordered.each do |subdir| %>
<li><%= link_to subdir.name, subdir %></li>
<% end %>
</ul>
</div>
<% end %> <% end %>
<div class="subdirectories">
<ul class="disc">
<% if dir.parent.id != Directory::ROOT %>
<li class="parent">
<%= link_to "Parent", dir.parent %>
</li>
<% end %>
<% dir.subdirs.ordered.each do |subdir| %>
<li><%= namelink subdir %></li>
<% end %>
</ul>
</div>
</div> </div>
<h3>Files</h3> <h3>Files</h3>
<% dir.files.unrelated.each do |file| %> <% dir.files.unrelated.each do |file| %>
@ -77,7 +83,7 @@
<% if cuser and cuser.admin? %> <% if cuser and cuser.admin? %>
<div class="controls"> <div class="controls">
<%= link_to "Edit Directory", edit_directory_path(dir), class: 'button' %> <%= 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 Directory", { controller: "directories", action: "new", id: dir }, { class: 'button' } %>
<%= link_to "New File", { controller: "data_files", action: "new", id: dir }, { class: 'button' } %> <%= link_to "New File", { controller: "data_files", action: "new", id: dir }, { class: 'button' } %>
</div> </div>

View file

@ -85,7 +85,11 @@ Ensl::Application.routes.draw do
resources :maps resources :maps
resources :logs resources :logs
resources :log_files resources :log_files
resources :directories resources :directories do
member do
get :recreate
end
end
resources :data_files resources :data_files
resources :predictions resources :predictions
resources :weeks resources :weeks