This repository has been archived on 2024-09-05. You can view files and clone it, but cannot push or open issues or pull requests.
shimmie2/ext/bulk_import_export/main.php

185 lines
6.4 KiB
PHP
Raw Normal View History

2021-12-14 18:32:47 +00:00
<?php
declare(strict_types=1);
2020-06-02 23:05:09 +00:00
namespace Shimmie2;
2020-06-02 23:05:09 +00:00
class BulkImportExport extends DataHandlerExtension
{
2021-12-14 18:32:47 +00:00
public const EXPORT_ACTION_NAME = "bulk_export";
public const EXPORT_INFO_FILE_NAME = "export.json";
protected array $SUPPORTED_MIME = [MimeType::ZIP];
2020-06-02 23:05:09 +00:00
public function onDataUpload(DataUploadEvent $event)
{
global $user, $database;
2020-06-02 23:05:09 +00:00
2020-06-14 16:05:55 +00:00
if ($this->supported_mime($event->mime) &&
2020-06-02 23:05:09 +00:00
$user->can(Permissions::BULK_IMPORT)) {
2023-01-11 11:15:26 +00:00
$zip = new \ZipArchive();
2020-06-02 23:05:09 +00:00
if ($zip->open($event->tmpname) === true) {
$json_data = $this->get_export_data($zip);
2020-07-07 16:07:23 +00:00
if (empty($json_data)) {
return;
2020-06-02 23:05:09 +00:00
}
2020-06-02 23:05:09 +00:00
$total = 0;
2020-06-04 14:48:43 +00:00
$skipped = 0;
$failed = 0;
2020-06-02 23:05:09 +00:00
$database->commit();
2020-06-02 23:05:09 +00:00
while (!empty($json_data)) {
$item = array_pop($json_data);
$database->begin_transaction();
try {
$image = Image::by_hash($item->hash);
2023-11-11 21:49:12 +00:00
if ($image != null) {
$skipped++;
2020-10-26 15:22:30 +00:00
log_info(BulkImportExportInfo::KEY, "Post $item->hash already present, skipping");
$database->commit();
continue;
}
2020-06-02 23:05:09 +00:00
$tmpfile = tempnam(sys_get_temp_dir(), "shimmie_bulk_import");
$stream = $zip->getStream($item->hash);
if ($zip === false) {
throw new SCoreException("Could not import " . $item->hash . ": File not in zip");
}
2020-06-02 23:05:09 +00:00
file_put_contents($tmpfile, $stream);
$id = add_image($tmpfile, $item->filename, $item->tags)->image_id;
2020-06-02 23:05:09 +00:00
2023-11-11 21:49:12 +00:00
if ($id == -1) {
2020-06-02 23:05:09 +00:00
throw new SCoreException("Unable to import file $item->hash");
}
$image = Image::by_id($id);
2023-11-11 21:49:12 +00:00
if ($image == null) {
2020-06-02 23:05:09 +00:00
throw new SCoreException("Unable to import file $item->hash");
}
2023-11-11 21:49:12 +00:00
if ($item->source != null) {
2020-06-02 23:05:09 +00:00
$image->set_source($item->source);
}
send_event(new BulkImportEvent($image, $item));
$database->commit();
2020-06-02 23:05:09 +00:00
$total++;
2023-01-11 11:15:26 +00:00
} catch (\Exception $ex) {
$failed++;
try {
$database->rollBack();
2023-01-11 11:15:26 +00:00
} catch (\Exception $ex2) {
log_error(BulkImportExportInfo::KEY, "Could not roll back transaction: " . $ex2->getMessage(), "Could not import " . $item->hash . ": " . $ex->getMessage());
}
log_error(BulkImportExportInfo::KEY, "Could not import " . $item->hash . ": " . $ex->getMessage(), "Could not import " . $item->hash . ": " . $ex->getMessage());
continue;
2020-06-02 23:05:09 +00:00
} finally {
if (!empty($tmpfile) && is_file($tmpfile)) {
2020-06-02 23:05:09 +00:00
unlink($tmpfile);
}
}
}
$event->image_id = -2; // default -1 = upload wasn't handled
log_info(
BulkImportExportInfo::KEY,
"Imported $total items, skipped $skipped, $failed failed",
"Imported $total items, skipped $skipped, $failed failed"
);
2020-06-02 23:05:09 +00:00
} else {
throw new SCoreException("Could not open zip archive");
}
}
}
public function onBulkActionBlockBuilding(BulkActionBlockBuildingEvent $event)
{
2020-10-29 01:28:46 +00:00
global $user;
2020-06-02 23:05:09 +00:00
if ($user->can(Permissions::BULK_EXPORT)) {
$event->add_action(self::EXPORT_ACTION_NAME, "Export");
}
}
public function onBulkAction(BulkActionEvent $event)
{
global $user, $page;
if ($user->can(Permissions::BULK_EXPORT) &&
($event->action == self::EXPORT_ACTION_NAME)) {
$download_filename = $user->name . '-' . date('YmdHis') . '.zip';
$zip_filename = tempnam(sys_get_temp_dir(), "shimmie_bulk_export");
2023-01-11 11:15:26 +00:00
$zip = new \ZipArchive();
2020-06-02 23:05:09 +00:00
$json_data = [];
2023-01-11 11:15:26 +00:00
if ($zip->open($zip_filename, \ZIPARCHIVE::CREATE | \ZIPARCHIVE::OVERWRITE) === true) {
2020-06-02 23:05:09 +00:00
foreach ($event->items as $image) {
$img_loc = warehouse_path(Image::IMAGE_DIR, $image->hash, false);
2023-02-04 20:50:26 +00:00
$export_event = send_event(new BulkExportEvent($image));
2020-06-02 23:05:09 +00:00
$data = $export_event->fields;
$data["hash"] = $image->hash;
$data["tags"] = $image->get_tag_array();
$data["filename"] = $image->filename;
$data["source"] = $image->source;
2023-01-11 11:15:26 +00:00
$json_data[] = $data;
2020-06-02 23:05:09 +00:00
$zip->addFile($img_loc, $image->hash);
}
$json_data = json_encode($json_data, JSON_PRETTY_PRINT);
$zip->addFromString(self::EXPORT_INFO_FILE_NAME, $json_data);
$zip->close();
$page->set_mode(PageMode::FILE);
$page->set_file($zip_filename, true);
$page->set_filename($download_filename);
$event->redirect = false;
2020-06-02 23:05:09 +00:00
}
}
}
// we don't actually do anything, just accept one upload and spawn several
protected function media_check_properties(MediaCheckPropertiesEvent $event): void
{
}
protected function check_contents(string $tmpname): bool
{
return false;
}
protected function create_thumb(string $hash, string $mime): bool
2020-06-02 23:05:09 +00:00
{
return false;
}
2023-01-11 11:15:26 +00:00
private function get_export_data(\ZipArchive $zip): ?array
{
$info = $zip->getStream(self::EXPORT_INFO_FILE_NAME);
if ($info !== false) {
try {
$json_string = stream_get_contents($info);
$json_data = json_decode($json_string);
2023-01-11 11:59:56 +00:00
return $json_data;
} finally {
fclose($info);
}
} else {
return null;
}
}
2020-06-02 23:05:09 +00:00
}