2021-08-24 19:59:30 +02:00
|
|
|
<?php
|
|
|
|
session_start();
|
2021-11-27 12:52:29 +01:00
|
|
|
|
|
|
|
/** require autoloading to manage namespaces */
|
2021-08-24 19:59:30 +02:00
|
|
|
require __DIR__ . '/vendor/autoload.php';
|
|
|
|
|
|
|
|
use PhotoPrismUpload\API\PhotoPrism;
|
2021-11-27 12:52:29 +01:00
|
|
|
use PhotoPrismUpload\Entities\Album;
|
2021-11-27 10:59:29 +01:00
|
|
|
|
2021-11-27 12:52:29 +01:00
|
|
|
/** @var string $footer Footer text which links to the Gitea repo */
|
2021-11-27 10:59:29 +01:00
|
|
|
$footer = '<footer style="position: fixed;bottom: 0;left: 0;">'
|
|
|
|
.'<a href="/git/phlaym/photoprismupload">Ich bin Open Source</a></footer>';
|
2021-08-24 19:59:30 +02:00
|
|
|
?>
|
|
|
|
<html>
|
|
|
|
<head>
|
2021-11-26 21:02:49 +01:00
|
|
|
<meta charset="UTF-8">
|
2021-11-26 20:57:46 +01:00
|
|
|
<meta name="color-scheme" content="dark light">
|
2021-11-26 21:02:49 +01:00
|
|
|
<title>Photoprism Upload</title>
|
|
|
|
<meta name="description" content="Eine Seite um Photos zur Photoprism Instanz hochzuladen">
|
|
|
|
<meta name="author" content="Max Nuding">
|
|
|
|
<meta http-equiv="robots" content="noindex,nofollow">
|
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
2021-11-26 21:15:31 +01:00
|
|
|
<meta http-equiv=”content-language” content=”de-de”/>
|
2021-08-26 07:47:56 +00:00
|
|
|
<style>
|
2021-11-26 20:57:46 +01:00
|
|
|
::root {
|
|
|
|
background-color: white;
|
|
|
|
color: black;
|
|
|
|
color-scheme: light dark;
|
|
|
|
}
|
|
|
|
@media screen and (prefers-color-scheme: dark) {
|
|
|
|
:root {
|
|
|
|
background-color: rgb(54, 54, 54);
|
|
|
|
color: white;
|
|
|
|
}
|
2021-11-26 21:15:31 +01:00
|
|
|
a {
|
|
|
|
color: rgb(105, 105, 242);
|
|
|
|
}
|
|
|
|
a:visited {
|
|
|
|
color: rgb(152, 95, 215);
|
|
|
|
}
|
2021-11-26 20:57:46 +01:00
|
|
|
}
|
2021-08-26 07:47:56 +00:00
|
|
|
.form-wrapper {
|
|
|
|
display: grid;
|
|
|
|
grid-template-rows: auto auto auto;
|
2021-11-27 08:01:15 +01:00
|
|
|
grid-auto-columns: minmax(auto, 300px) auto;
|
2021-08-26 07:47:56 +00:00
|
|
|
}
|
|
|
|
label[for="album"] {
|
|
|
|
grid-column: 1;
|
|
|
|
grid-row: 1;
|
2021-11-27 09:24:10 +01:00
|
|
|
align-self: center;
|
2021-08-26 07:47:56 +00:00
|
|
|
}
|
|
|
|
#album {
|
|
|
|
grid-column: 2;
|
|
|
|
grid-row: 1;
|
|
|
|
}
|
|
|
|
#input {
|
|
|
|
grid-row: 2;
|
|
|
|
grid-column: 1/3;
|
|
|
|
}
|
|
|
|
.form-wrapper > form:nth-child(1) {
|
|
|
|
display: inherit;
|
|
|
|
}
|
|
|
|
input[type=submit] {
|
2021-11-26 20:43:53 +01:00
|
|
|
grid-column: 2;
|
|
|
|
grid-row: 3;
|
|
|
|
justify-self: right;
|
|
|
|
}
|
2021-11-27 12:52:29 +01:00
|
|
|
#error,
|
|
|
|
#fileProgress,
|
|
|
|
#totalProgress,
|
|
|
|
label[for="fileProgress"],
|
|
|
|
label[for="totalProgress"] {
|
2021-11-26 20:43:53 +01:00
|
|
|
display:none;
|
|
|
|
grid-column: 1;
|
|
|
|
}
|
|
|
|
#uploadForm {
|
|
|
|
grid-row: 1;
|
|
|
|
grid-column: 1;
|
|
|
|
}
|
|
|
|
#error {
|
|
|
|
grid-row: 2;
|
2021-11-27 08:01:15 +01:00
|
|
|
grid-column: 1/3;
|
2021-11-26 20:43:53 +01:00
|
|
|
}
|
|
|
|
label[for="fileProgress"] {
|
2021-08-26 07:47:56 +00:00
|
|
|
grid-row: 3;
|
2021-11-26 20:43:53 +01:00
|
|
|
}
|
|
|
|
#fileProgress {
|
|
|
|
grid-row: 4;
|
|
|
|
width: 100%;
|
|
|
|
}
|
|
|
|
label[for="totalProgress"]{
|
|
|
|
grid-row: 5;
|
|
|
|
}
|
|
|
|
#totalProgress {
|
|
|
|
grid-row: 6;
|
|
|
|
width: 100%;
|
2021-08-26 07:47:56 +00:00
|
|
|
}
|
2021-11-27 10:51:30 +01:00
|
|
|
#viewAlbum {
|
2021-11-27 12:52:29 +01:00
|
|
|
grid-row: 7;
|
2021-11-27 10:51:30 +01:00
|
|
|
}
|
2021-11-26 21:15:31 +01:00
|
|
|
footer {
|
|
|
|
margin: 8px;
|
|
|
|
}
|
2021-08-26 07:47:56 +00:00
|
|
|
</style>
|
2021-08-24 19:59:30 +02:00
|
|
|
</head>
|
2021-11-26 17:44:38 +01:00
|
|
|
<body>
|
2021-08-24 19:59:30 +02:00
|
|
|
<?php
|
|
|
|
|
2021-11-27 12:52:29 +01:00
|
|
|
/** @var array $config configuration options */
|
2021-08-26 07:47:56 +00:00
|
|
|
$config = require(__DIR__ . '/config.php');
|
2021-11-27 12:52:29 +01:00
|
|
|
|
|
|
|
/** @var PhotoPrism $api API object to interface with PhotoPrism */
|
2021-08-26 07:47:56 +00:00
|
|
|
$api = new PhotoPrism($config);
|
2021-11-27 12:52:29 +01:00
|
|
|
|
|
|
|
/** @var Album[] $albums List of PhotoPrism albums */
|
2021-08-26 07:47:56 +00:00
|
|
|
$albums = [];
|
|
|
|
try {
|
|
|
|
$api->login();
|
|
|
|
} catch (\Exception $e) {
|
2021-11-27 08:55:20 +01:00
|
|
|
die('Fehler: ' . $e->getMessage().$footer.'</body></html>');
|
2021-08-26 07:47:56 +00:00
|
|
|
}
|
|
|
|
|
2021-08-24 19:59:30 +02:00
|
|
|
if (!isset($_POST['submit'])) {
|
2021-08-26 10:56:59 +02:00
|
|
|
if (!isset($_GET['token'])) {
|
2021-11-27 08:55:20 +01:00
|
|
|
die('Sorry, kein Zugriff' . $footer . '</body></html>');
|
2021-08-26 10:56:59 +02:00
|
|
|
}
|
2021-11-27 12:52:29 +01:00
|
|
|
|
|
|
|
/** @var string $token Tokens for which album(s) are visible in the dropdown */
|
2021-08-26 10:56:59 +02:00
|
|
|
$token = $_GET['token'];
|
2021-11-27 12:52:29 +01:00
|
|
|
|
|
|
|
/** @var string[] $tokens List of album tokens */
|
2021-08-26 10:56:59 +02:00
|
|
|
$tokens = explode(',', $token);
|
2021-11-27 12:52:29 +01:00
|
|
|
|
|
|
|
/** @var string $album_url URL path to the selected album */
|
|
|
|
$album_url = '/';
|
2021-08-26 10:56:59 +02:00
|
|
|
try {
|
|
|
|
$albums = $api->getAlbumsByTokens($tokens);
|
|
|
|
} catch (\Exception $e) {
|
2021-11-27 08:55:20 +01:00
|
|
|
die('Fehler: ' . $footer . $e->getMessage() . '</body></html>');
|
2021-08-26 10:56:59 +02:00
|
|
|
}
|
|
|
|
if (empty($albums) && (empty($config['noAlbumToken']) || !in_array($config['noAlbumToken'], $tokens))) {
|
2021-11-27 08:55:20 +01:00
|
|
|
die('Falscher Token' . $footer . '</body></html>');
|
2021-08-26 10:56:59 +02:00
|
|
|
}
|
2021-08-24 19:59:30 +02:00
|
|
|
?>
|
2021-08-26 07:47:56 +00:00
|
|
|
<div class="form-wrapper">
|
2021-11-26 20:43:53 +01:00
|
|
|
<form method="POST" enctype="multipart/form-data" id="uploadForm">
|
2021-08-26 07:47:56 +00:00
|
|
|
<label for="album">Zu Album hinzufügen</label>
|
|
|
|
<select name="album" id="album">
|
2021-11-27 10:51:30 +01:00
|
|
|
<option value="" data-url="/">---</option>
|
2021-11-27 12:52:29 +01:00
|
|
|
<?php
|
|
|
|
/** @var Album $album Current PhotoPrism albums */
|
|
|
|
foreach ($albums as $album) {
|
|
|
|
/** @var string $selected Selected attribute of the option */
|
|
|
|
$selected = $album->token === $token ? ' selected ' : '';
|
2021-11-27 10:51:30 +01:00
|
|
|
if ($album->token === $token) {
|
2021-11-27 12:52:29 +01:00
|
|
|
$album_url = $album->getUrlPath() ?? '/';
|
2021-11-27 10:51:30 +01:00
|
|
|
}
|
|
|
|
echo '<option value="'
|
|
|
|
. $album->uid
|
|
|
|
. '"'
|
|
|
|
. $selected
|
|
|
|
. 'data-url='
|
|
|
|
. ($album->getUrlPath() ?? '/')
|
|
|
|
. '>'
|
|
|
|
. $album->title
|
|
|
|
. '</option>\n';
|
|
|
|
}
|
|
|
|
$album_url = "https://photos.phlaym.net{$album_url}";
|
|
|
|
?>
|
2021-08-26 07:47:56 +00:00
|
|
|
</select>
|
2021-08-26 10:56:59 +02:00
|
|
|
<input multiple type="file" name="files[]" id="input" required/>
|
2021-08-26 07:47:56 +00:00
|
|
|
<input type="submit" name="submit" value="Upload" />
|
|
|
|
</form>
|
2021-11-26 20:43:53 +01:00
|
|
|
<div id="error"></div>
|
|
|
|
<label for="fileProgress">Datei:</label>
|
|
|
|
<progress id="fileProgress"></progress>
|
|
|
|
<label for="totalProgress">Gesamt:</label>
|
|
|
|
<progress max="0" value="0" id="totalProgress"></progress>
|
2021-11-27 10:51:30 +01:00
|
|
|
<a href="<?=$album_url;?>" target="_blank" id="viewAlbum">Album ansehen</a>
|
2021-08-26 07:47:56 +00:00
|
|
|
</div>
|
2021-08-24 19:59:30 +02:00
|
|
|
<script>
|
2021-11-27 08:01:15 +01:00
|
|
|
window.tooLarge = false;
|
|
|
|
window.tooManyFiles = false;
|
2021-11-26 20:43:53 +01:00
|
|
|
const form = document.getElementById('uploadForm');
|
|
|
|
const submitButton = form.querySelector('input[type=submit]');
|
|
|
|
const albumInput = form.querySelector('select[name=album]');
|
|
|
|
const input = document.getElementById('input');
|
|
|
|
const fileProgress = document.getElementById('fileProgress');
|
|
|
|
const totalProgress = document.getElementById('totalProgress');
|
|
|
|
const fileProgressLabel = document.querySelector('label[for=fileProgress]');
|
|
|
|
const totalProgressLabel = document.querySelector('label[for=totalProgress]');
|
|
|
|
const errorDiv = document.getElementById('error');
|
2021-11-27 10:51:30 +01:00
|
|
|
const albumAnchor = document.getElementById('viewAlbum');
|
2021-11-26 20:43:53 +01:00
|
|
|
|
|
|
|
async function postData(url, data = {}, method = 'POST') {
|
|
|
|
const response = await fetch(url, {
|
|
|
|
method: method,
|
|
|
|
body: data
|
|
|
|
});
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
2021-11-27 13:05:26 +01:00
|
|
|
function validateFileType(file) {
|
|
|
|
if (file.type && (file.type.startsWith('image/') || file.type.startsWith('video/'))) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
const parts = file.name.split('.');
|
|
|
|
const extension = parts.length > 0 ? parts[parts.length-1] : '';
|
|
|
|
if (['jpg', 'jpeg', 'png', 'heic', 'heif', 'mov', 'mp4', 'mkv'].includes(extension)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
console.warn('Invalid file type', extension);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-11-27 10:51:30 +01:00
|
|
|
albumInput.addEventListener('change', (event) => {
|
|
|
|
console.log(event);
|
|
|
|
albumAnchor.href = `https://photos.phlaym.net${albumInput.selectedOptions[0].dataset.url}`;
|
|
|
|
});
|
|
|
|
|
2021-11-26 20:43:53 +01:00
|
|
|
form.addEventListener('submit', async function(event) {
|
|
|
|
event.preventDefault();
|
2021-11-27 08:01:15 +01:00
|
|
|
const isInvalid = window.tooLarge || window.tooManyFiles;
|
|
|
|
if (isInvalid) {
|
|
|
|
console.error('Aborting upload! Too many fiels or fiels too large');
|
|
|
|
return;
|
|
|
|
}
|
2021-11-26 20:43:53 +01:00
|
|
|
errorDiv.innerText = '';
|
|
|
|
|
|
|
|
fileProgressLabel.style.display = 'inherit';
|
|
|
|
totalProgressLabel.style.display = 'inherit';
|
|
|
|
fileProgress.style.display = 'inherit';
|
|
|
|
totalProgress.style.display = 'inherit';
|
|
|
|
|
|
|
|
let idx = 0;
|
|
|
|
for (file of fileList) {
|
|
|
|
console.log('Starting upload', file);
|
|
|
|
|
|
|
|
fileProgressLabel.innerText = `Datei: ${file.name}`
|
|
|
|
totalProgressLabel.innerText = `Gesamt: ${idx} von ${fileList.length} fertig`;
|
|
|
|
totalProgress.value = idx++;
|
|
|
|
|
|
|
|
let formData = new FormData();
|
|
|
|
formData.set(input.name, file);
|
|
|
|
formData.set(submitButton.name, submitButton.value);
|
|
|
|
formData.set(albumInput.name, albumInput.value);
|
|
|
|
try {
|
|
|
|
let resp = await postData(form.action, formData, form.method);
|
|
|
|
} catch(e) {
|
|
|
|
console.error('Error uploading file', e);
|
|
|
|
errorDiv.innerHTML += `Fehler beim Upload der Datei ${file.name}: ${e}<br />`;
|
|
|
|
errorDiv.style.display = 'block';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
totalProgressLabel.innerText = `Gesamt: ${idx} von ${fileList.length} fertig`;
|
|
|
|
totalProgress.value = idx++;
|
|
|
|
fileProgress.max = 1;
|
|
|
|
fileProgress.value = 1;
|
|
|
|
});
|
|
|
|
|
2021-08-24 19:59:30 +02:00
|
|
|
|
2021-11-26 20:43:53 +01:00
|
|
|
let fileList = [];
|
2021-08-24 19:59:30 +02:00
|
|
|
input.addEventListener('change', (event) => {
|
2021-11-27 08:01:15 +01:00
|
|
|
const maxFileSize = <?=$config['fileUploadLimitMb'];?>;
|
|
|
|
const maxAmountOfFiles = <?=$config['maximumNumberOfFilesPerUpload'];?>;
|
2021-08-24 19:59:30 +02:00
|
|
|
const errorDiv = document.getElementById('error');
|
2021-11-26 20:43:53 +01:00
|
|
|
const totalProgress = document.getElementById('totalProgress');
|
2021-11-27 08:01:15 +01:00
|
|
|
|
2021-08-24 19:59:30 +02:00
|
|
|
errorDiv.innerText = '';
|
|
|
|
errorDiv.style.display = 'none';
|
2021-11-27 08:01:15 +01:00
|
|
|
submitButton.disabled = false;
|
2021-08-24 19:59:30 +02:00
|
|
|
|
|
|
|
const target = event.target;
|
2021-11-26 20:43:53 +01:00
|
|
|
fileList = [];
|
2021-11-27 08:01:15 +01:00
|
|
|
const filesTooLarge = [];
|
2021-08-24 19:59:30 +02:00
|
|
|
if (target.files) {
|
|
|
|
for (file of target.files) {
|
2021-11-27 08:01:15 +01:00
|
|
|
const sizeInMb = file.size / 1024 / 1024;
|
|
|
|
if (sizeInMb >= maxFileSize) {
|
|
|
|
filesTooLarge.push(file.name);
|
2021-11-27 10:51:30 +01:00
|
|
|
console.warn(
|
|
|
|
'File',
|
|
|
|
file.name,
|
|
|
|
'is',
|
|
|
|
sizeInMb,
|
|
|
|
'MB big, which is over the limit of',
|
|
|
|
maxFileSize);
|
2021-11-27 13:05:26 +01:00
|
|
|
} else if(validateFileType(file)) {
|
|
|
|
fileList.push(file);
|
2021-11-27 08:01:15 +01:00
|
|
|
}
|
2021-11-27 13:05:26 +01:00
|
|
|
|
2021-08-24 19:59:30 +02:00
|
|
|
}
|
|
|
|
}
|
2021-11-26 20:43:53 +01:00
|
|
|
totalProgress.max = fileList.length;
|
2021-11-27 08:01:15 +01:00
|
|
|
|
|
|
|
window.tooManyFiles = fileList.length > maxAmountOfFiles;
|
|
|
|
if (window.tooManyFiles) {
|
|
|
|
errorDiv.style.display = 'block';
|
2021-11-27 10:59:29 +01:00
|
|
|
errorDiv.innerHTML += ```
|
|
|
|
Das sind zu viele Dateien, du darfst max.
|
|
|
|
${maxAmountOfFiles} Dateien gleichzeitig hochladen. ```;
|
2021-11-27 08:01:15 +01:00
|
|
|
submitButton.disabled = true;
|
|
|
|
console.warn('Total files:', target.files.length, '. Too many!');
|
|
|
|
}
|
|
|
|
|
|
|
|
window.tooLarge = filesTooLarge.length > 0;
|
|
|
|
if (window.tooLarge) {
|
|
|
|
const names = filesTooLarge.join(', ')
|
|
|
|
errorDiv.style.display = 'block';
|
2021-11-27 10:59:29 +01:00
|
|
|
const pluralizedMessage = filesTooLarge.length > 1
|
|
|
|
? 'Die folgenden Dateien sind'
|
|
|
|
: 'Die folgende Datei ist';
|
|
|
|
errorDiv.innerHTML += ```
|
2021-11-27 13:05:26 +01:00
|
|
|
${pluralizedMessage} zu groß und wird beim Upload ignoriert: ${names}.
|
2021-11-27 10:59:29 +01:00
|
|
|
Jede Datei darf max. ${maxFileSize} MB groß sein.```;
|
2021-11-27 13:05:26 +01:00
|
|
|
}
|
|
|
|
if (!fileList.length) {
|
2021-11-27 08:01:15 +01:00
|
|
|
submitButton.disabled = true;
|
2021-11-27 13:08:28 +01:00
|
|
|
errorDiv.style.display = 'block';
|
2021-11-27 13:05:26 +01:00
|
|
|
errorDiv.innerHTML += 'Keine gültigen Bilder oder Videos gefunden';
|
2021-11-27 08:01:15 +01:00
|
|
|
}
|
2021-08-24 19:59:30 +02:00
|
|
|
});
|
|
|
|
</script>
|
|
|
|
<?php
|
2021-11-27 08:55:20 +01:00
|
|
|
die($footer . '</body></html>');
|
2021-08-24 19:59:30 +02:00
|
|
|
}
|
|
|
|
try {
|
2021-08-26 07:47:56 +00:00
|
|
|
$api->uploadPhotos($_POST['album']);
|
2021-08-24 19:59:30 +02:00
|
|
|
} catch (\Exception $e) {
|
2021-11-27 08:55:20 +01:00
|
|
|
die('Fehler: ' . $footer . $e->getMessage() .'</body></html>');
|
2021-08-24 19:59:30 +02:00
|
|
|
}
|
2021-08-26 07:47:56 +00:00
|
|
|
?>
|
2021-11-26 17:44:38 +01:00
|
|
|
Erfolg! <a href=".">Zurück</a>
|
2021-11-26 20:43:53 +01:00
|
|
|
</body></html>
|