Кафедра ИСиТ УО ВГТУ
  • Специальности
    • Экономика электронного бизнеса
    • Информационные системы
    • Information Control Systems
  • Каталог
  • Сайт кафедры
  • Сервисы
    • GitLab
    • ownCloud
    • JupyterHub
    • JupyterHub 2
    • VNC
    • Soft
  1. ИСиТ
  2. РВПсИПП
  3. Практика
  4. Лаб. работа 6 “Работа с файлами (эмуляция S3-хранилища)”
  • ИСиТ
    • АОС
      • Теория
        • Введение в операционные системы
        • Управление памятью
        • Управление процессами
        • Система ввода-вывода
        • Информационная безопасность
        • Виртуализация
      • Практика
    • РВПсИПП
      • Теория
        • Настройка среды разработки для PHP
        • Разработка web-приложений на базе фреймворков
        • Основы Laravel
        • Шаблоны в Laravel
        • Модели и базы данных в Laravel
        • Формы и валидация в Laravel
        • Аутентификация и авторизация в Laravel
        • Создание REST API в Laravel
        • Работа с файлами и изображениями в Laravel
        • Тестирование и отладка в Laravel
        • Введение в фреймворк Symfony
        • Маршруты и контроллеры в Symfony
        • Шаблоны и Twig в Symfony
        • Формы и валидация в Symfony
        • Доступ к базам данных в Symfony
        • Аутентификация и авторизация в Symfony
        • Сервисы и зависимости в Symfony
        • Создание REST API в Symfony
        • Работа с файлами и медиа в Symfony
        • Сравнение и выбор фреймворка
        • Развертывание веб-приложения
      • Практика
        • Лаб. работа 1 “Создание нового приложения Laravel”
        • Лаб. работа 2 “Добавление главной страницы и базовых маршрутов”
        • Лаб. работа 3 “Создание моделей, миграций и сидеров”
        • Лаб. работа 4 “Создание индексных страниц и пагинация”
        • Лаб. работа 5 “Создание форм для работы с сущностями”
        • Лаб. работа 6 “Работа с файлами (эмуляция S3-хранилища)”
        • Лаб. работа “Создание маршрутов в Laravel”
        • Лаб. работа “Работа с базами данных в Laravel”
        • Лаб. работа “Работа с формами в Laravel”
        • Лаб. работа “Аутентификация и авторизация в Laravel”
        • Лаб. работа “Работа с файлами в Laravel”
        • Лаб. работа “Тестирование и оптимизация в Laravel”
        • Лаб. работа “Создание REST API в Laravel”
        • Лаб. работа “Основы Symfony”
        • Лаб. работа “Шаблоны и представления в Symfony”
        • Лаб. работа “Работа с базами данных в Symfony”
        • Лаб. работа “Фомы и аутентификация в Symfony”
        • Лаб. работа “Сервисы и зависимости в Symfony”
        • Лаб. работа “REST API в Symfony”
        • Лаб. работа “Работа с медиа контентом в Symfony”
        • Лаб. работа “Создание и развертывание проекта”
    • ПСП
      • Теория
        • Введение
        • Протокол HTTP
        • Программирование с использованием сокетов
        • Введение в PHP
        • Работа с базами данных в PHP
        • Объектно-ориентированные возможности PHP
        • Настройка среды разработки для PHP
        • Разработка web-приложений на базе фреймворков
      • Практика
        • Программное обеспечение
        • Регистрация в JupyterHub
        • Лаб. работа “Почтовые протоколы”
        • Лаб. работа “Протокол FTP”
        • Лаб. работа “Протокол HTTP”
        • Лаб. работа “Программирование сетевых приложений с использованием сокетов”
        • Лаб. работа “Основы PHP”
        • Лаб. работа “Массивы в PHP”
        • Лаб. работа “Создание веб-приложений с использованием Slim”
    • Компьютерные сети
      • Теория
        • Введение в компьютерные сети
        • Топологии сетей
        • Кодирование и мультиплексирование
        • Стеки протоколов
        • Адресация в компьютерных сетях
        • Система доменных имен (DNS)
        • Программирование с использованием сокетов
        • Введение в PHP
        • Протокол HTTP
        • Введение в компьютерные сети
      • Практика
        • Программное обеспечение
        • Регистрация в JupyterHub
        • Лаб. работа “Почтовые протоколы”
        • Лаб. работа “Протокол FTP”
        • Лаб. работа “Протокол HTTP”
        • Лаб. работа “Программирование сетевых приложений с использованием сокетов”
        • Лаб. работа “Основы PHP”
        • Лаб работа “Массивы в PHP”
    • РиОИИС
      • Теория
        • Классификация оптимизационных задач
        • Генетические алгоритмы
        • Системы массового обслуживания
        • Теория игр
        • Машинное обучение
        • Глубокое обучение (Deep learning)
        • Основы функционального программирования
        • Основы программирования на Haskell
        • Введение в логическое программирование
        • Инференция и рассуждения в логическом программировании
        • Разработка экспертных систем
        • Интеллектуальные системы и их архитектура
        • Веб-скрэйпинг
        • Сбор данных с открытых API
      • Практика
        • JupyterHub
        • Лаб. работа “Методы одномерной оптимизации”
        • Лаб. работа “Методы многомерной оптимизации”
        • Лаб. работа “Функции в Python”
        • Лаб. работа “Рекурсия в Python”
        • Лаб. работа “Итераторы в Python”
        • Лаб. работа “Генетические алгоритмы”
        • Лаб. работа “Haskell”
        • Лаб. работа “Логическое программирование”
        • Лаб. работа “Сбор данных с помощью веб-скрейпинга”
    • КСКР
      • Практика
        • Лаб. работа “Одномерные и двумерные массивы в C#”
        • Лаб. работа “Обращение матриц в C#”
    • Системное программирование
      • Теория
        • Управление памятью в Windows
        • Файловые операции в Windows
        • Управление процессами в Windows
        • Графический интерфейс Windows
        • ОС Unix
      • Практика
        • Лаб. работа “Работа с динамической памятью в Windows”
        • Лаб. работа “Операции с файлами в Windows”
        • Лаб. работа “Управление процессами в Windows”
        • Лаб. работа “Работа с виртуальной машиной Linux”
        • Лаб. работа “Язык командного энтерпритатора Shell”
        • Лаб. работа “Работа с файлами в Linux”
        • Лаб. работа “Работа с процессами в Linux”

Содержание

  • Задачи
  • Методическая часть
  • Индивидуальные задания
  • Проверка работы
  • Примечания
  1. ИСиТ
  2. РВПсИПП
  3. Практика
  4. Лаб. работа 6 “Работа с файлами (эмуляция S3-хранилища)”

Лаб. работа 6 “Работа с файлами (эмуляция S3-хранилища)”

Разработка web-приложений с использованием программных платформ
Практика
Автор

Бизюк Андрей

Дата публикации

27 февраля 2025 г.

Тема: Реализация загрузки, хранения и управления файлами с использованием локального хранилища в качестве эмуляции S3.
Цель: Научиться работать с файловой системой Laravel, имитируя облачное хранилище, и реализовать CRUD для мультимедийных данных.


Задачи

  1. Настроить фейковый S3-диск для локальной разработки.
  2. Реализовать загрузку файлов к постам.
  3. Добавить валидацию и оптимизацию файлов.
  4. Организовать просмотр и удаление файлов.

Методическая часть

1. Настройка файлового хранилища

  1. Добавьте новый диск в config/filesystems.php:

    'disks' => [
        // ... другие диски
        's3-fake' => [
            'driver' => 'local',
            'root' => storage_path('app/s3-fake'),
            'url' => env('APP_URL') . '/storage/s3-fake',
            'visibility' => 'public',
        ],
    ],
  2. Обновите .env:

    FILESYSTEM_DISK=s3-fake
  3. Создайте директорию и симлинк:

    mkdir -p storage/app/s3-fake
    php artisan storage:link

2. Модификация модели и миграции

  1. Создайте модель Media:

    php artisan make:model Media -m
  2. Добавьте поля в миграцию:

    Schema::create('media', function (Blueprint $table) {
        $table->id();
        $table->string('path');
        $table->foreignId('post_id')->constrained()->cascadeOnDelete();
        $table->string('type'); // image, video, document
        $table->timestamps();
    });
  3. Добавьте связь в модели:

    • В Post.php:

      public function media() {
          return $this->hasMany(Media::class);
      }

3. Обновление контроллера PostController

  1. Метод store для загрузки файлов:

    use Illuminate\Support\Facades\Storage;
    
    public function store(Request $request) {
        $validated = $request->validate([
            'title' => 'required|max:255',
            'files.*' => 'file|mimes:jpg,png,pdf,mp4|max:5120',
        ]);
    
        $post = Post::create($validated);
    
        if ($request->hasFile('files')) {
            foreach ($request->file('files') as $file) {
                $path = $file->store("posts/{$post->id}", 's3-fake');
                Media::create([
                    'post_id' => $post->id,
                    'path' => $path,
                    'type' => $file->getMimeType(),
                ]);
            }
        }
    
        return redirect()->route('posts.index');
    }
  2. Метод destroyFile для удаления:

    public function destroyFile(Media $media) {
        Storage::disk('s3-fake')->delete($media->path);
        $media->delete();
        return back()->with('success', 'Файл удалён!');
    }

4. Blade-шаблоны

  1. Форма загрузки файлов (posts/create.blade.php):

    <form method="POST" action="{{ route('posts.store') }}" enctype="multipart/form-data">
        @csrf
        <div class="mb-4">
            <label>Прикрепить файлы:</label>
            <input type="file" name="files[]" multiple class="border p-2 w-full">
            @error('files.*') <div class="text-red-500">{{ $message }}</div> @enderror
        </div>
        <!-- Остальные поля формы -->
    </form>
  2. Галерея файлов (posts/show.blade.php):

    <div class="grid grid-cols-3 gap-4">
        @foreach ($post->media as $media)
            <div class="relative">
                @if (Str::startsWith($media->type, 'image'))
                    <img src="{{ Storage::disk('s3-fake')->url($media->path) }}" 
                         class="w-full h-48 object-cover">
                @else
                    <div class="bg-gray-100 p-4">
                        <p>{{ basename($media->path) }}</p>
                    </div>
                @endif
                <form method="POST" action="{{ route('posts.deleteFile', $media) }}" 
                      class="absolute top-2 right-2">
                    @csrf @method('DELETE')
                    <button type="submit" class="bg-red-500 text-white p-1 rounded">×</button>
                </form>
            </div>
        @endforeach
    </div>

Индивидуальные задания

1. Валидация имен файлов

  • Запретите загрузку файлов с пробелами в имени:

    Validator::extend('no_spaces', function ($attr, $value, $params) {
        return !preg_match('/\s/', $value->getClientOriginalName());
    });
    // Использование:
    'files.*' => 'no_spaces',

2. Логирование операций

  • Создайте middleware LogFileOperations:

    public function handle($request, Closure $next) {
        if ($request->hasFile('files')) {
            \Log::info("Uploaded files: " . json_encode($request->file('files')));
        }
        return $next($request);
    }
  • Зарегистрируйте его в bootstrap/app.php:

    ->withMiddleware(function (Middleware $middleware) {
        $middleware->append(\App\Http\Middleware\LogFileOperations::class);
    })

3. Оптимизация изображений (опционально)

  • Установите пакет:

    composer require spatie/image-optimizer
  • Оптимизируйте изображения перед сохранением:

    use Spatie\ImageOptimizer\OptimizerChainFactory;
    // ...
    foreach ($request->file('files') as $file) {
        if (Str::startsWith($file->getMimeType(), 'image')) {
            $optimizer = OptimizerChainFactory::create();
            $optimizer->optimize($file->getRealPath());
        }
        // Сохранение файла...
    }

Проверка работы

  1. Загрузка файлов:
    • Откройте форму создания поста.
    • Загрузите несколько файлов (изображения, PDF).
    • Убедитесь, что файлы сохранились в storage/app/s3-fake/posts/{post_id}.
  2. Просмотр файлов:
    • Откройте страницу поста.
    • Проверьте, что изображения отображаются, а PDF показывается как ссылка.
  3. Удаление файлов:
    • Нажмите кнопку удаления файла.
    • Убедитесь, что файл удалён из хранилища и БД.

Примечания

  • Преимущества подхода:
    • Эмуляция облачного хранилища экономит ресурсы.
    • Легко переключиться на реальный S3 через .env.
  • Важно:
    • Для продакшена замените s3-fake на s3 в .env.
    • Для тестирования используйте Storage::fake('s3-fake').
Наверх
Лаб. работа 5 “Создание форм для работы с сущностями”
Лаб. работа “Создание маршрутов в Laravel”