2020-06-02 18:05:09 -05:00
|
|
|
<?php declare(strict_types=1);
|
|
|
|
|
|
|
|
|
|
|
|
class BulkImportExport extends DataHandlerExtension
|
|
|
|
{
|
|
|
|
const EXPORT_ACTION_NAME = "bulk_export";
|
|
|
|
const EXPORT_INFO_FILE_NAME = "export.json";
|
2020-06-14 11:05:55 -05:00
|
|
|
protected $SUPPORTED_MIME = [MimeType::ZIP];
|
2020-06-02 18:05:09 -05:00
|
|
|
|
|
|
|
|
|
|
|
public function onDataUpload(DataUploadEvent $event)
|
|
|
|
{
|
2020-06-12 13:39:57 -05:00
|
|
|
global $user, $database;
|
2020-06-02 18:05:09 -05:00
|
|
|
|
2020-06-14 11:05:55 -05:00
|
|
|
if ($this->supported_mime($event->mime) &&
|
2020-06-02 18:05:09 -05:00
|
|
|
$user->can(Permissions::BULK_IMPORT)) {
|
|
|
|
$zip = new ZipArchive;
|
|
|
|
|
|
|
|
if ($zip->open($event->tmpname) === true) {
|
2020-06-24 08:41:01 -05:00
|
|
|
$json_data = $this->get_export_data($zip);
|
|
|
|
|
2020-07-07 16:07:23 +00:00
|
|
|
if (empty($json_data)) {
|
2020-06-24 08:41:01 -05:00
|
|
|
return;
|
2020-06-02 18:05:09 -05:00
|
|
|
}
|
2020-06-24 08:41:01 -05:00
|
|
|
|
2020-06-02 18:05:09 -05:00
|
|
|
$total = 0;
|
2020-06-04 09:48:43 -05:00
|
|
|
$skipped = 0;
|
2020-06-16 18:31:33 -05:00
|
|
|
$failed = 0;
|
2020-06-02 18:05:09 -05:00
|
|
|
|
2020-06-12 13:39:57 -05:00
|
|
|
$database->commit();
|
|
|
|
|
2020-06-02 18:05:09 -05:00
|
|
|
while (!empty($json_data)) {
|
|
|
|
$item = array_pop($json_data);
|
2020-06-12 13:39:57 -05:00
|
|
|
$database->begin_transaction();
|
|
|
|
try {
|
|
|
|
$image = Image::by_hash($item->hash);
|
|
|
|
if ($image!=null) {
|
|
|
|
$skipped++;
|
|
|
|
log_info(BulkImportExportInfo::KEY, "Image $item->hash already present, skipping");
|
|
|
|
$database->commit();
|
|
|
|
continue;
|
|
|
|
}
|
2020-06-02 18:05:09 -05:00
|
|
|
|
2020-06-16 18:31:33 -05:00
|
|
|
$tmpfile = tempnam(sys_get_temp_dir(), "shimmie_bulk_import");
|
2020-06-12 13:39:57 -05:00
|
|
|
$stream = $zip->getStream($item->hash);
|
|
|
|
if ($zip === false) {
|
|
|
|
throw new SCoreException("Could not import " . $item->hash . ": File not in zip");
|
|
|
|
}
|
2020-06-02 18:05:09 -05:00
|
|
|
|
|
|
|
file_put_contents($tmpfile, $stream);
|
|
|
|
|
|
|
|
$id = add_image($tmpfile, $item->filename, Tag::implode($item->tags));
|
|
|
|
|
|
|
|
if ($id==-1) {
|
|
|
|
throw new SCoreException("Unable to import file $item->hash");
|
|
|
|
}
|
|
|
|
|
|
|
|
$image = Image::by_id($id);
|
|
|
|
|
|
|
|
if ($image==null) {
|
|
|
|
throw new SCoreException("Unable to import file $item->hash");
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($item->source!=null) {
|
|
|
|
$image->set_source($item->source);
|
|
|
|
}
|
|
|
|
send_event(new BulkImportEvent($image, $item));
|
|
|
|
|
2020-06-12 13:39:57 -05:00
|
|
|
$database->commit();
|
2020-06-02 18:05:09 -05:00
|
|
|
$total++;
|
|
|
|
} catch (Exception $ex) {
|
2020-06-16 18:31:33 -05:00
|
|
|
$failed++;
|
2020-06-12 13:39:57 -05:00
|
|
|
try {
|
|
|
|
$database->rollBack();
|
|
|
|
} 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 18:05:09 -05:00
|
|
|
} finally {
|
2020-06-12 13:39:57 -05:00
|
|
|
if (!empty($tmpfile) && is_file($tmpfile)) {
|
2020-06-02 18:05:09 -05:00
|
|
|
unlink($tmpfile);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$event->image_id = -2; // default -1 = upload wasn't handled
|
|
|
|
|
2020-06-16 18:31:33 -05:00
|
|
|
log_info(
|
|
|
|
BulkImportExportInfo::KEY,
|
|
|
|
"Imported $total items, skipped $skipped, $failed failed",
|
|
|
|
"Imported $total items, skipped $skipped, $failed failed"
|
|
|
|
);
|
2020-06-02 18:05:09 -05:00
|
|
|
} else {
|
|
|
|
throw new SCoreException("Could not open zip archive");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public function onBulkActionBlockBuilding(BulkActionBlockBuildingEvent $event)
|
|
|
|
{
|
|
|
|
global $user, $config;
|
|
|
|
|
|
|
|
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)) {
|
2020-06-16 18:31:33 -05:00
|
|
|
$download_filename = $user->name . '-' . date('YmdHis') . '.zip';
|
|
|
|
$zip_filename = tempnam(sys_get_temp_dir(), "shimmie_bulk_export");
|
2020-06-02 18:05:09 -05:00
|
|
|
$zip = new ZipArchive;
|
|
|
|
|
|
|
|
$json_data = [];
|
|
|
|
|
|
|
|
if ($zip->open($zip_filename, ZIPARCHIVE::CREATE | ZIPARCHIVE::OVERWRITE) === true) {
|
|
|
|
foreach ($event->items as $image) {
|
|
|
|
$img_loc = warehouse_path(Image::IMAGE_DIR, $image->hash, false);
|
|
|
|
|
|
|
|
$export_event = new BulkExportEvent($image);
|
|
|
|
send_event($export_event);
|
|
|
|
$data = $export_event->fields;
|
|
|
|
$data["hash"] = $image->hash;
|
|
|
|
$data["tags"] = $image->get_tag_array();
|
|
|
|
$data["filename"] = $image->filename;
|
|
|
|
$data["source"] = $image->source;
|
|
|
|
|
|
|
|
array_push($json_data, $data);
|
|
|
|
|
|
|
|
$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);
|
2020-06-16 18:31:33 -05:00
|
|
|
$page->set_filename($download_filename);
|
|
|
|
|
|
|
|
$event->redirect = false;
|
2020-06-02 18:05:09 -05: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 $type): bool
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2020-06-24 08:41:01 -05:00
|
|
|
|
|
|
|
private function get_export_data(ZipArchive $zip): ?array
|
|
|
|
{
|
|
|
|
$info = $zip->getStream(self::EXPORT_INFO_FILE_NAME);
|
|
|
|
$json_data = [];
|
|
|
|
if ($info !== false) {
|
|
|
|
try {
|
|
|
|
$json_string = stream_get_contents($info);
|
|
|
|
$json_data = json_decode($json_string);
|
|
|
|
} finally {
|
|
|
|
fclose($info);
|
|
|
|
}
|
|
|
|
return $json_data;
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
2020-06-02 18:05:09 -05:00
|
|
|
}
|