aboutsummaryrefslogtreecommitdiffstats
path: root/Zotlabs/Thumbs/Epubthumb.php
blob: af372e85c41912f51481be1bccb217a2fbd0e853 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
<?php

namespace Zotlabs\Thumbs;

use DOMDocument;
use DOMElement;
use DOMXPath;
use GdImage;
use ZipArchive;

/**
 * Thumbnail creation for epub files.
 */
class Epubthumb {

	/**
	 * Match for application/epub+zip.
	 *
	 * @param string $type MimeType
	 * @return boolean
	 */
	function Match(string $type): bool {
		return $type === 'application/epub+zip';
	}

	/**
	 * Create the thumbnail if the Epub has a cover.
	 *
	 * @param array $attach
	 * @param int $preview_style unused
	 * @param int $height (optional) default 300
	 * @param int $width (optional) default 300
	 *
	 * @SuppressWarnings(PHPMD.UnusedFormalParameter)
	 * phpcs:disable Generic.CodeAnalysis.UnusedFunctionParameter.FoundBeforeLastUsed
	 */
	function Thumb($attach, $preview_style, $height = 300, $width = 300): void {

		$file = dbunescbin($attach['content']);
		if (!$file) {
			return;
		}

		$image = $this->getCoverFromEpub($file);

		if ($image) {
			$srcwidth = imagesx($image);
			$srcheight = imagesy($image);

			$dest = imagecreatetruecolor($width, $height);
			imagealphablending($dest, false);
			imagesavealpha($dest, true);

			imagecopyresampled($dest, $image, 0, 0, 0, 0, $width, $height, $srcwidth, $srcheight);

			imagejpeg($dest, "{$file}.thumb");

			imagedestroy($image);
			imagedestroy($dest);
		}
	}

	/**
	 * Fetch the cover from the epub archive, if it's present.
	 *
	 * There's a few limitations here: This will only work if the cover
	 * is a raster image of a supported format. SVG does not work, neither
	 * will other schemes sometimes used for cover/front page.
	 *
	 * @param string $filename	The local filename of the epub archive.
	 *
	 * @return GdImage|false	If a cover is found, it is returned as a
	 *	GdImage object. Otherwise return false.
	 */
	private function getCoverFromEpub(string $filename): GdImage|false {
		$epub = new ZipArchive();
		$rc = $epub->open($filename, ZipArchive::RDONLY);

		if ($rc !== true) {
			logger("Error opening file '{$filename}': rc = ${rc}.", LOGGER_DEBUG, LOG_DEBUG);
			return false;
		}

		$cover = false;
		$cover_name = $this->parseEpub($epub);
		if ($cover_name !== false) {
			$cover = $epub->getFromName($cover_name);
			if ($cover === false) {
				logger("File '{$cover_name}' not found in EPUB.", LOGGER_DEBUG, LOG_DEBUG);
			}
		}

		$epub->close();

		if ($cover !== false && !empty($cover)) {
			return imagecreatefromstring($cover);
		} else {
			return false;
		}
	}

	/**
	 * Parse the epub to find the path of the cover image.
	 *
	 * @param ZipArchive $epub	An opened epub ZipArchive.
	 *
	 * @return string|false		The path to the cover image or false.
	 */
	private function parseEpub(ZipArchive $epub): string|false {
		$packagePath = $this->getEpubPackagePath($epub);
		if ($packagePath !== false) {
			$package = $epub->getFromName($packagePath);
			if ($package === false || empty($package)) {
				logger("Package file '${packagePath}' not found in EPUB", LOGGER_DEBUG, LOG_DEBUG);
				return false;
			}

			$domdoc = new DOMDocument();
			$domdoc->loadXML($package);
			$xpath = new DOMXPath($domdoc);
			$xpath->registerNamespace("n", "http://www.idpf.org/2007/opf");
			$nodes = $xpath->query('/n:package/n:manifest/n:item[@properties="cover-image"]');

			if ($nodes->count() === 0) {
				logger('No cover found in EPUB manifest.', LOGGER_DEBUG, LOG_DEBUG);
				return false;
			}

			$node = $nodes->item(0);
			if ($node === null) {
				logger('No nodes in non-empty node list?', LOGGER_DEBUG, LOG_DEBUG);
				return false;
			}

			if (is_a($node, DOMElement::class)) {
				// The URL's in the package file is relative to the subdirectory
				// within the epub archive where it is located. See
				// https://www.w3.org/TR/epub-33/#sec-parsing-urls-metainf
				return dirname($packagePath) . '/' . $node->getAttribute('href');
			}
		}

		return false;
	}

	/**
	 * Locate the package file within the epub.
	 *
	 * The package file in an epub archive contains the manifest
	 * that again may contain a reference to the cover for the
	 * epub.
	 *
	 * @param ZipArchive $epub	An opened epub archive.
	 *
	 * @return string|false		The full pathname of the package file or false.
	 */
	private function getEpubPackagePath(ZipArchive $epub): string|false {
		//
		// The only mandatory known file within the archive is the
		// container file, so we fetch it to find the reference to
		// the package file.
		//
		// See: https://www.w3.org/TR/epub-33/#sec-container-metainf
		//
		$container = $epub->getFromName('META-INF/container.xml');

		if ($container === false || empty($container)) {
			logger('No container in archive, probably not an EPUB.', LOGGER_DEBUG, LOG_DEBUG);
			return false;
		}

		$domdoc = new DOMDocument();
		$domdoc->loadXML($container);
		$nodes = $domdoc->getElementsByTagName('rootfile');

		if ($nodes->count() == 0) {
			logger('EPUB rootfile not found, is this an epub?', LOGGER_DEBUG, LOG_DEBUG);
			return false;
		}

		$packageNode = $nodes->item(0);
		if ($packageNode === null || !is_a($packageNode, DOMElement::class)) {
			logger('EPUB rootfile element missing or invalid.', LOGGER_DEBUG, LOG_DEBUG);
			return false;
		}

		$packagePath = $packageNode->getAttribute('full-path');
		$packageMediaType = $packageNode->getAttribute('media-type');

		if (empty($packagePath) || $packageMediaType !== 'application/oebps-package+xml') {
			logger('EPUB package path missing or incorrect media type.', LOGGER_DEBUG, LOG_DEBUG);
			return false;
		}

		return $packagePath;
	}
}