mirror of
https://0xacab.org/sutty/sutty
synced 2024-11-24 08:36:21 +00:00
feat: testear subida de archivos #14433
This commit is contained in:
parent
79a08d5dd3
commit
036f64f958
2 changed files with 218 additions and 45 deletions
|
@ -4,19 +4,25 @@ require 'filemagic'
|
||||||
|
|
||||||
# Define un campo de archivo
|
# Define un campo de archivo
|
||||||
class MetadataFile < MetadataTemplate
|
class MetadataFile < MetadataTemplate
|
||||||
|
include Metadata::NonIndexableConcern
|
||||||
|
include Metadata::AlwaysPublicConcern
|
||||||
|
|
||||||
# Una ruta vacía a la imagen con una descripción vacía
|
# Una ruta vacía a la imagen con una descripción vacía
|
||||||
def default_value
|
def default_value
|
||||||
super || { 'path' => nil, 'description' => nil }
|
super || { 'path' => nil, 'description' => '' }
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# La descripción es opcional
|
||||||
|
#
|
||||||
|
# @return [Boolean]
|
||||||
def empty?
|
def empty?
|
||||||
value == default_value
|
value.nil? || value['path'].blank?
|
||||||
end
|
end
|
||||||
|
|
||||||
# No hay valores sugeridos para archivos subidos.
|
# No hay valores sugeridos para archivos subidos.
|
||||||
#
|
def values
|
||||||
# XXX: Esto ayuda a deserializar en {Site#everything_of}
|
raise NotImplementedError, "#{self.class} no tiene valores sugeridos"
|
||||||
def values; end
|
end
|
||||||
|
|
||||||
def validate
|
def validate
|
||||||
super
|
super
|
||||||
|
@ -29,25 +35,15 @@ class MetadataFile < MetadataTemplate
|
||||||
errors.empty?
|
errors.empty?
|
||||||
end
|
end
|
||||||
|
|
||||||
# Determina si necesitamos la imagen pero no la tenemos
|
|
||||||
def path_missing?
|
|
||||||
required && !path?
|
|
||||||
end
|
|
||||||
|
|
||||||
# Determina si el archivo ya fue subido
|
|
||||||
def uploaded?
|
|
||||||
value['path'].is_a?(String)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Asociar la imagen subida al sitio y obtener la ruta
|
# Asociar la imagen subida al sitio y obtener la ruta
|
||||||
# @return [Boolean]
|
# @return [Boolean]
|
||||||
def save
|
def save
|
||||||
if value['path'].blank?
|
return true unless changed?
|
||||||
self[:value] = default_value
|
|
||||||
else
|
self[:value] = default_value if empty?
|
||||||
value['description'] = sanitize value['description']
|
self[:value] = sanitize(value)
|
||||||
value['path'] = relative_destination_path_with_filename.to_s if static_file
|
|
||||||
end
|
value['path'] = relative_destination_path_with_filename.to_s if static_file
|
||||||
|
|
||||||
true
|
true
|
||||||
end
|
end
|
||||||
|
@ -75,32 +71,20 @@ class MetadataFile < MetadataTemplate
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Obtiene la ruta absoluta al archivo
|
|
||||||
#
|
|
||||||
# @return [Pathname]
|
|
||||||
def pathname
|
|
||||||
raise NoMethodError unless uploaded?
|
|
||||||
|
|
||||||
@pathname ||= Pathname.new(File.join(site.path, value['path']))
|
|
||||||
end
|
|
||||||
|
|
||||||
# Obtiene la key del attachment a partir de la ruta
|
|
||||||
#
|
|
||||||
# @return [String]
|
|
||||||
def key_from_path
|
|
||||||
@key_from_path ||= pathname.dirname.basename.to_s
|
|
||||||
end
|
|
||||||
|
|
||||||
def path?
|
|
||||||
value['path'].present?
|
|
||||||
end
|
|
||||||
|
|
||||||
def description?
|
|
||||||
value['description'].present?
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
# Valida que estemos pasando el formato correcto
|
||||||
|
#
|
||||||
|
# @param :value [Any]
|
||||||
|
# @return [Hash,nil]
|
||||||
|
def sanitize(value)
|
||||||
|
return unless value.is_a? Hash
|
||||||
|
|
||||||
|
value.dup.tap do |v|
|
||||||
|
v['description'] = super(v['description'])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
# Obtener la ruta al archivo relativa al sitio
|
# Obtener la ruta al archivo relativa al sitio
|
||||||
#
|
#
|
||||||
# @return [Pathname]
|
# @return [Pathname]
|
||||||
|
@ -135,6 +119,10 @@ class MetadataFile < MetadataTemplate
|
||||||
value['path']
|
value['path']
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# La ruta absoluta al archivo
|
||||||
|
#
|
||||||
|
# @todo Eliminar retrocompatibilidad
|
||||||
|
# @return [String]
|
||||||
def static_file_path
|
def static_file_path
|
||||||
case static_file.blob.service.name
|
case static_file.blob.service.name
|
||||||
when :local
|
when :local
|
||||||
|
@ -171,4 +159,39 @@ class MetadataFile < MetadataTemplate
|
||||||
ExceptionNotifier.notify_exception(e, data: { site: site.name, path: value['path'] })
|
ExceptionNotifier.notify_exception(e, data: { site: site.name, path: value['path'] })
|
||||||
nil
|
nil
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Determina si necesitamos la imagen pero no la tenemos
|
||||||
|
def path_missing?
|
||||||
|
required && !path?
|
||||||
|
end
|
||||||
|
|
||||||
|
# Determina si el archivo ya fue subido
|
||||||
|
def uploaded?
|
||||||
|
value['path'].is_a?(String)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Obtiene la ruta absoluta al archivo
|
||||||
|
#
|
||||||
|
# @return [Pathname]
|
||||||
|
def pathname
|
||||||
|
raise NoMethodError unless uploaded?
|
||||||
|
|
||||||
|
@pathname ||= Pathname.new(File.join(site.path, value['path']))
|
||||||
|
end
|
||||||
|
|
||||||
|
# Obtiene la key del attachment a partir de la ruta
|
||||||
|
#
|
||||||
|
# @return [String]
|
||||||
|
def key_from_path
|
||||||
|
@key_from_path ||= pathname.dirname.basename.to_s
|
||||||
|
end
|
||||||
|
|
||||||
|
# @todo Este método no se puede correr sobre archivos recién subidos
|
||||||
|
def path?
|
||||||
|
value['path'].present?
|
||||||
|
end
|
||||||
|
|
||||||
|
def description?
|
||||||
|
value['description'].present?
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
150
test/models/metadata_file_test.rb
Normal file
150
test/models/metadata_file_test.rb
Normal file
|
@ -0,0 +1,150 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'test_helper'
|
||||||
|
|
||||||
|
class MetadataFileTest < ActiveSupport::TestCase
|
||||||
|
setup do
|
||||||
|
@site = build(:site)
|
||||||
|
@document = build(:document, site: @site.jekyll)
|
||||||
|
@name = SecureRandom.hex
|
||||||
|
@layout = build(:layout, site: @site)
|
||||||
|
@metadata = MetadataFile.new(site: @site, document: @document, name: @name, type: 'file', layout: @layout)
|
||||||
|
end
|
||||||
|
|
||||||
|
def random_value
|
||||||
|
{
|
||||||
|
'path' => "public/#{SecureRandom.base36(28)}/#{SecureRandom.hex}.pdf",
|
||||||
|
'description' => SecureRandom.hex
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'se guarda en el encabezado' do
|
||||||
|
assert @metadata.front_matter?
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'por defecto es una ruta vacía' do
|
||||||
|
assert_equal ({ 'path' => nil, 'description' => '' }), @metadata.default_value
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'puede traer el valor desde el documento' do
|
||||||
|
@document.data[@name] = random_value
|
||||||
|
|
||||||
|
assert_equal @document[@name], @metadata.document_value
|
||||||
|
assert @metadata.save
|
||||||
|
assert_equal @document[@name], @metadata.value
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'al cambiar el valor podemos obtenerlo' do
|
||||||
|
@metadata.value = value_was = random_value
|
||||||
|
@metadata.value = random_value
|
||||||
|
|
||||||
|
assert_equal value_was, @metadata.value_was
|
||||||
|
assert_not_equal value_was, @metadata.value
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'pueden tener un valor por defecto desde el layout' do
|
||||||
|
default = random_value
|
||||||
|
|
||||||
|
@layout = build(:layout, site: @site, metadata: { @name => { 'default' => { I18n.locale.to_s => default } } })
|
||||||
|
@metadata = MetadataFile.new(site: @site, document: @document, name: @name, type: 'file', layout: @layout)
|
||||||
|
|
||||||
|
assert_equal default, @metadata.default_value
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'no puede estar vacía si es obligatoria' do
|
||||||
|
@layout = build(:layout, site: @site, metadata: { @name => { 'default' => { 'required' => true } } })
|
||||||
|
@metadata = MetadataFile.new(site: @site, document: @document, name: @name, type: 'file', layout: @layout,
|
||||||
|
required: true)
|
||||||
|
|
||||||
|
assert @metadata.required
|
||||||
|
assert @metadata.empty?
|
||||||
|
assert_not @metadata.valid?
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'no se pueden indexar' do
|
||||||
|
assert_not @metadata.indexable?
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'son públicos por defecto' do
|
||||||
|
assert_not @metadata.private?
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'no se pueden hacer privados' do
|
||||||
|
@layout = build(:layout, site: @site, metadata: { @name => { 'private' => true } })
|
||||||
|
@metadata = MetadataFile.new(site: @site, document: @document, name: @name, type: 'file', layout: @layout)
|
||||||
|
|
||||||
|
assert_not @metadata.private?
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'se les puede asignar un valor' do
|
||||||
|
@metadata.value = value = random_value
|
||||||
|
|
||||||
|
assert_equal value, @metadata.value
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'se lo considera vacío si no se asigna un archivo' do
|
||||||
|
assert @metadata.empty?
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'se puede subir un archivo' do
|
||||||
|
value = random_value
|
||||||
|
value['path'] =
|
||||||
|
ActionDispatch::Http::UploadedFile.new(tempfile: Tempfile.new, filename: value['path'].split('/').last)
|
||||||
|
|
||||||
|
# XXX: por alguna razón no se guarda esto en cascada
|
||||||
|
@site.design.save
|
||||||
|
@site.design_id = @site.design.id
|
||||||
|
@site.save
|
||||||
|
|
||||||
|
@metadata.value = value
|
||||||
|
|
||||||
|
assert @metadata.valid?
|
||||||
|
assert @metadata.save
|
||||||
|
assert_instance_of String, @metadata.value['path']
|
||||||
|
assert_instance_of ActiveStorage::Attachment, @metadata.static_file
|
||||||
|
assert_not @metadata.empty?
|
||||||
|
ensure
|
||||||
|
@site.destroy
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'se puede volver a encontrar un archivo subido' do
|
||||||
|
value = random_value
|
||||||
|
value['path'] =
|
||||||
|
ActionDispatch::Http::UploadedFile.new(tempfile: Tempfile.new, filename: value['path'].split('/').last)
|
||||||
|
|
||||||
|
@site.design.save
|
||||||
|
@site.design_id = @site.design.id
|
||||||
|
@site.save
|
||||||
|
|
||||||
|
@metadata.value = value
|
||||||
|
@metadata.save
|
||||||
|
|
||||||
|
@metadata.instance_variable_set :@static_file, nil
|
||||||
|
|
||||||
|
assert_instance_of String, @metadata.value['path']
|
||||||
|
assert_instance_of ActiveStorage::Attachment, @metadata.static_file
|
||||||
|
assert_not @metadata.empty?
|
||||||
|
ensure
|
||||||
|
@site.destroy
|
||||||
|
end
|
||||||
|
|
||||||
|
test 'si el archivo esta en el repositorio lo podemos asociar al sitio' do
|
||||||
|
value = random_value
|
||||||
|
|
||||||
|
@site.design.save
|
||||||
|
@site.design_id = @site.design.id
|
||||||
|
@site.save
|
||||||
|
|
||||||
|
path = File.join(@site.path, value['path'])
|
||||||
|
FileUtils.mkdir_p File.dirname(path)
|
||||||
|
|
||||||
|
File.write(path, 'test')
|
||||||
|
|
||||||
|
@metadata.value = value
|
||||||
|
@metadata.save
|
||||||
|
|
||||||
|
assert_instance_of ActiveStorage::Attachment, @metadata.static_file
|
||||||
|
ensure
|
||||||
|
@site.destroy
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in a new issue