2018-02-26 04:03:07 -08:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
|
|
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
|
|
|
|
//
|
|
|
|
// Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
// purpose with or without fee is hereby granted, provided that the above
|
|
|
|
// copyright notice and this permission notice is present in all copies. The
|
|
|
|
// full license for this software is available in the LICENSE file.
|
|
|
|
|
2018-03-08 12:38:02 -08:00
|
|
|
#include "media.h"
|
2018-02-26 04:03:07 -08:00
|
|
|
|
2019-05-29 19:20:17 -07:00
|
|
|
using namespace ircd::m::media::thumbnail; //TODO: XXX
|
2019-08-21 23:57:07 -07:00
|
|
|
using namespace ircd;
|
2019-05-29 19:20:17 -07:00
|
|
|
|
|
|
|
decltype(ircd::m::media::thumbnail::enable)
|
|
|
|
ircd::m::media::thumbnail::enable
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.media.thumbnail.enable" },
|
|
|
|
{ "default", true },
|
|
|
|
};
|
|
|
|
|
2019-07-25 21:38:35 -07:00
|
|
|
decltype(ircd::m::media::thumbnail::enable_remote)
|
|
|
|
ircd::m::media::thumbnail::enable_remote
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.media.thumbnail.enable_remote" },
|
|
|
|
{ "default", true },
|
|
|
|
};
|
|
|
|
|
2021-01-21 21:51:28 -08:00
|
|
|
decltype(ircd::m::media::thumbnail::animation_enable)
|
|
|
|
ircd::m::media::thumbnail::animation_enable
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.media.thumbnail.animation.enable" },
|
|
|
|
{ "default", true },
|
|
|
|
};
|
|
|
|
|
2019-05-29 19:20:17 -07:00
|
|
|
decltype(ircd::m::media::thumbnail::width_min)
|
|
|
|
ircd::m::media::thumbnail::width_min
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.media.thumbnail.width.min" },
|
2019-05-30 00:17:45 -07:00
|
|
|
{ "default", 16L },
|
2019-05-29 19:20:17 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
decltype(ircd::m::media::thumbnail::width_max)
|
|
|
|
ircd::m::media::thumbnail::width_max
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.media.thumbnail.width.max" },
|
|
|
|
{ "default", 1536L },
|
|
|
|
};
|
|
|
|
|
|
|
|
decltype(ircd::m::media::thumbnail::height_min)
|
|
|
|
ircd::m::media::thumbnail::height_min
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.media.thumbnail.height.min" },
|
2019-05-30 00:17:45 -07:00
|
|
|
{ "default", 16L },
|
2019-05-29 19:20:17 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
decltype(ircd::m::media::thumbnail::height_max)
|
|
|
|
ircd::m::media::thumbnail::height_max
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.media.thumbnail.height.max" },
|
|
|
|
{ "default", 1536L },
|
|
|
|
};
|
|
|
|
|
2019-07-05 18:20:39 -07:00
|
|
|
decltype(ircd::m::media::thumbnail::mime_whitelist)
|
|
|
|
ircd::m::media::thumbnail::mime_whitelist
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.media.thumbnail.mime.whitelist" },
|
|
|
|
{ "default", "image/jpeg image/png image/webp" },
|
|
|
|
};
|
|
|
|
|
|
|
|
decltype(ircd::m::media::thumbnail::mime_blacklist)
|
|
|
|
ircd::m::media::thumbnail::mime_blacklist
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.media.thumbnail.mime.blacklist" },
|
|
|
|
{ "default", "" },
|
|
|
|
};
|
|
|
|
|
2019-09-28 16:12:07 -07:00
|
|
|
m::resource
|
2018-02-26 04:03:07 -08:00
|
|
|
thumbnail_resource
|
|
|
|
{
|
|
|
|
"/_matrix/media/r0/thumbnail/",
|
|
|
|
{
|
2018-04-12 14:44:30 -07:00
|
|
|
"(11.7.1.4) thumbnails",
|
2018-02-26 04:03:07 -08:00
|
|
|
resource::DIRECTORY,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-09-28 16:12:07 -07:00
|
|
|
static m::resource::response
|
2018-03-08 12:38:02 -08:00
|
|
|
get__thumbnail_local(client &client,
|
2019-09-28 16:12:07 -07:00
|
|
|
const m::resource::request &request,
|
2019-08-21 23:57:07 -07:00
|
|
|
const m::media::mxc &,
|
2019-05-30 00:52:36 -07:00
|
|
|
const m::room &room);
|
2018-02-26 04:03:07 -08:00
|
|
|
|
2019-09-28 16:12:07 -07:00
|
|
|
m::resource::response
|
2018-02-26 04:03:07 -08:00
|
|
|
get__thumbnail(client &client,
|
2019-09-28 16:12:07 -07:00
|
|
|
const m::resource::request &request)
|
2018-02-26 04:03:07 -08:00
|
|
|
{
|
|
|
|
if(request.parv.size() < 1)
|
2019-06-15 02:01:47 -06:00
|
|
|
throw m::NEED_MORE_PARAMS
|
2018-02-26 04:03:07 -08:00
|
|
|
{
|
2019-06-15 02:01:47 -06:00
|
|
|
"Server name parameter required"
|
2018-02-26 04:03:07 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
if(request.parv.size() < 2)
|
2019-06-15 02:01:47 -06:00
|
|
|
throw m::NEED_MORE_PARAMS
|
2018-02-26 04:03:07 -08:00
|
|
|
{
|
2019-06-15 02:01:47 -06:00
|
|
|
"Media ID parameter required"
|
2018-02-26 04:03:07 -08:00
|
|
|
};
|
|
|
|
|
2023-02-22 13:14:46 -08:00
|
|
|
char url_buf[2][256];
|
2019-08-21 23:57:07 -07:00
|
|
|
const m::media::mxc mxc
|
2018-02-26 04:03:07 -08:00
|
|
|
{
|
2023-02-22 13:14:46 -08:00
|
|
|
url::decode(url_buf[0], request.parv[0]),
|
|
|
|
url::decode(url_buf[1], request.parv[1]),
|
2018-02-26 04:03:07 -08:00
|
|
|
};
|
|
|
|
|
2018-04-30 07:30:34 -07:00
|
|
|
// Thumbnail doesn't require auth so if there is no user_id detected
|
|
|
|
// then we download on behalf of @ircd.
|
|
|
|
const m::user::id &user_id
|
|
|
|
{
|
|
|
|
request.user_id?
|
|
|
|
m::user::id{request.user_id}:
|
2019-09-30 20:50:58 -07:00
|
|
|
m::me()
|
2018-04-30 07:30:34 -07:00
|
|
|
};
|
|
|
|
|
2019-07-25 21:38:35 -07:00
|
|
|
if(!m::media::thumbnail::enable_remote)
|
|
|
|
{
|
|
|
|
const m::room::id::buf room_id
|
|
|
|
{
|
2019-08-21 23:57:07 -07:00
|
|
|
m::media::file::room_id(mxc)
|
2019-07-25 21:38:35 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
if(!exists(room_id))
|
2019-09-28 16:12:07 -07:00
|
|
|
return m::resource::response
|
2019-07-25 21:38:35 -07:00
|
|
|
{
|
|
|
|
client, http::NOT_FOUND
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-03-08 12:38:02 -08:00
|
|
|
const m::room::id::buf room_id
|
2018-02-26 04:03:07 -08:00
|
|
|
{
|
2019-08-21 23:57:07 -07:00
|
|
|
m::media::file::download(mxc, user_id)
|
2018-04-22 20:46:14 -07:00
|
|
|
};
|
|
|
|
|
2019-08-21 23:57:07 -07:00
|
|
|
return get__thumbnail_local(client, request, mxc, room_id);
|
2018-02-26 04:03:07 -08:00
|
|
|
}
|
|
|
|
|
2019-09-28 16:12:07 -07:00
|
|
|
static m::resource::method
|
2018-02-26 04:03:07 -08:00
|
|
|
method_get
|
|
|
|
{
|
2020-05-18 15:39:41 -07:00
|
|
|
thumbnail_resource,
|
|
|
|
"GET",
|
|
|
|
get__thumbnail,
|
|
|
|
{
|
|
|
|
m::resource::method::flag(0), // flag
|
|
|
|
45s, // timeout
|
|
|
|
}
|
2018-02-26 04:03:07 -08:00
|
|
|
};
|
|
|
|
|
2019-09-28 16:12:07 -07:00
|
|
|
static m::resource::response
|
2018-03-08 12:38:02 -08:00
|
|
|
get__thumbnail_local(client &client,
|
2019-09-28 16:12:07 -07:00
|
|
|
const m::resource::request &request,
|
2019-08-21 23:57:07 -07:00
|
|
|
const m::media::mxc &mxc,
|
2019-05-30 00:52:36 -07:00
|
|
|
const m::room &room)
|
2018-03-08 12:38:02 -08:00
|
|
|
{
|
2019-05-30 00:52:36 -07:00
|
|
|
const auto &method
|
|
|
|
{
|
|
|
|
request.query.get("method", "scale"_sv)
|
|
|
|
};
|
|
|
|
|
2020-07-25 01:53:23 -07:00
|
|
|
const size_t _dimension[]
|
2019-05-30 00:52:36 -07:00
|
|
|
{
|
|
|
|
request.query.get<size_t>("width", 0),
|
2020-07-25 01:53:23 -07:00
|
|
|
request.query.get<size_t>("height", 0),
|
2019-05-30 00:52:36 -07:00
|
|
|
};
|
|
|
|
|
2020-07-25 01:53:23 -07:00
|
|
|
const pair<size_t> dimension
|
2019-05-30 00:52:36 -07:00
|
|
|
{
|
2020-07-25 01:53:23 -07:00
|
|
|
_dimension[0]?
|
|
|
|
std::clamp(_dimension[0], size_t(width_min), size_t(width_max)):
|
|
|
|
_dimension[0],
|
2019-05-30 00:52:36 -07:00
|
|
|
|
2020-07-25 01:53:23 -07:00
|
|
|
_dimension[1]?
|
|
|
|
std::clamp(_dimension[1], size_t(height_min), size_t(height_max)):
|
|
|
|
_dimension[1]
|
|
|
|
};
|
2019-05-30 00:52:36 -07:00
|
|
|
|
2018-12-29 17:33:51 -08:00
|
|
|
static const m::event::fetch::opts fopts
|
|
|
|
{
|
|
|
|
m::event::keys::include {"content"}
|
|
|
|
};
|
|
|
|
|
|
|
|
const m::room::state state
|
|
|
|
{
|
|
|
|
room, &fopts
|
|
|
|
};
|
|
|
|
|
2018-03-08 12:38:02 -08:00
|
|
|
// Get the file's total size
|
|
|
|
size_t file_size{0};
|
2020-08-21 03:37:10 -07:00
|
|
|
state.get("ircd.file.stat.size", "", [&file_size]
|
2018-03-08 12:38:02 -08:00
|
|
|
(const m::event &event)
|
|
|
|
{
|
2020-08-21 03:37:10 -07:00
|
|
|
file_size = at<"content"_>(event).get<size_t>("bytes");
|
2018-03-08 12:38:02 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
// Get the MIME type
|
|
|
|
char type_buf[64];
|
2018-04-12 15:58:30 -07:00
|
|
|
string_view content_type
|
|
|
|
{
|
|
|
|
"application/octet-stream"
|
|
|
|
};
|
|
|
|
|
2020-08-21 03:37:10 -07:00
|
|
|
state.get("ircd.file.stat.type", "", [&type_buf, &content_type]
|
2018-03-08 12:38:02 -08:00
|
|
|
(const m::event &event)
|
|
|
|
{
|
2020-08-21 03:37:10 -07:00
|
|
|
const json::string &value
|
2018-03-08 12:38:02 -08:00
|
|
|
{
|
2020-08-21 03:37:10 -07:00
|
|
|
at<"content"_>(event).at("mime_type")
|
2018-03-08 12:38:02 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
content_type =
|
|
|
|
{
|
|
|
|
type_buf, copy(type_buf, value)
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
2019-05-29 17:44:10 -07:00
|
|
|
const unique_buffer<mutable_buffer> buf
|
2018-03-08 12:38:02 -08:00
|
|
|
{
|
2019-05-29 17:44:10 -07:00
|
|
|
file_size
|
2018-03-08 12:38:02 -08:00
|
|
|
};
|
|
|
|
|
2019-05-29 17:44:10 -07:00
|
|
|
size_t copied(0);
|
|
|
|
const auto sink{[&buf, &copied]
|
2018-04-24 18:26:35 -07:00
|
|
|
(const const_buffer &block)
|
2018-03-08 12:38:02 -08:00
|
|
|
{
|
2019-05-29 17:44:10 -07:00
|
|
|
copied += copy(buf + copied, block);
|
2018-04-22 20:44:15 -07:00
|
|
|
}};
|
|
|
|
|
2018-04-25 18:44:02 -07:00
|
|
|
const size_t read_size
|
2018-04-22 20:44:15 -07:00
|
|
|
{
|
2019-08-21 23:57:07 -07:00
|
|
|
m::media::file::read(room, sink)
|
2018-04-22 20:44:15 -07:00
|
|
|
};
|
2018-03-08 12:38:02 -08:00
|
|
|
|
2019-05-29 17:44:10 -07:00
|
|
|
if(unlikely(read_size != file_size || file_size != copied))
|
|
|
|
throw ircd::error
|
|
|
|
{
|
|
|
|
"File %s/%s [%s] size mismatch: expected %zu got %zu copied %zu",
|
2019-08-21 23:57:07 -07:00
|
|
|
mxc.server,
|
|
|
|
mxc.mediaid,
|
2019-05-29 17:44:10 -07:00
|
|
|
string_view{room.room_id},
|
|
|
|
file_size,
|
|
|
|
read_size,
|
|
|
|
copied
|
|
|
|
};
|
|
|
|
|
2019-07-05 18:20:39 -07:00
|
|
|
const auto mime_type
|
|
|
|
{
|
|
|
|
split(content_type, ';').first
|
|
|
|
};
|
|
|
|
|
2021-02-09 23:01:10 -08:00
|
|
|
const bool supported
|
|
|
|
{
|
|
|
|
// Available in build
|
2022-06-22 18:59:33 -07:00
|
|
|
IRCD_USE_MAGICK
|
2021-02-09 23:01:10 -08:00
|
|
|
|
|
|
|
// Enabled by configuration
|
|
|
|
&& enable
|
|
|
|
};
|
|
|
|
|
2019-07-05 18:20:39 -07:00
|
|
|
const bool permitted
|
|
|
|
{
|
|
|
|
// If there's a blacklist, mime type must not in the blacklist.
|
|
|
|
(!mime_blacklist || !has(mime_blacklist, mime_type))
|
|
|
|
|
|
|
|
// If there's a whitelist, mime type must be in the whitelist.
|
|
|
|
&& (!mime_whitelist || has(mime_whitelist, mime_type))
|
|
|
|
};
|
|
|
|
|
2021-01-21 21:51:28 -08:00
|
|
|
const bool animated
|
|
|
|
{
|
|
|
|
// Administrator's fuse to disable animation detection.
|
|
|
|
bool(animation_enable)
|
|
|
|
|
2021-02-09 23:01:10 -08:00
|
|
|
// Only call into libpng if magick is supported/enabled
|
|
|
|
&& supported
|
|
|
|
|
2021-01-21 21:51:28 -08:00
|
|
|
// If the type is not permitted don't bother checking for animation.
|
|
|
|
&& permitted
|
|
|
|
|
|
|
|
// Case for APNG; do not permit them to be thumbnailed
|
|
|
|
&& (has(mime_type, "image/png") && png::is_animated(buf))
|
|
|
|
};
|
|
|
|
|
2019-07-05 18:20:39 -07:00
|
|
|
const bool valid_args
|
|
|
|
{
|
|
|
|
// Both dimension parameters given in query string
|
|
|
|
(dimension.first && dimension.second)
|
|
|
|
|
|
|
|
// Known thumbnailing method in query string
|
|
|
|
&& (method == "scale" || method == "crop")
|
|
|
|
};
|
|
|
|
|
2019-05-30 00:52:36 -07:00
|
|
|
const bool fallback // Reasons to just send the original image
|
|
|
|
{
|
2021-02-09 23:01:10 -08:00
|
|
|
// Thumbnailer support not enabled or available
|
|
|
|
!supported
|
2019-06-04 02:22:09 -07:00
|
|
|
|
2019-07-05 18:20:39 -07:00
|
|
|
// Access denied for this operation
|
|
|
|
|| !permitted
|
2019-06-04 02:22:09 -07:00
|
|
|
|
2021-01-21 21:51:28 -08:00
|
|
|
// Bypassed to prevent loss of animation
|
|
|
|
|| animated
|
|
|
|
|
2019-07-05 18:20:39 -07:00
|
|
|
// Arguments invalid.
|
|
|
|
|| !valid_args
|
2019-05-30 00:52:36 -07:00
|
|
|
};
|
|
|
|
|
2020-08-24 01:40:46 -07:00
|
|
|
if(fallback && enable)
|
2019-07-05 18:20:39 -07:00
|
|
|
log::dwarning
|
|
|
|
{
|
|
|
|
"Not thumbnailing %s/%s [%s] '%s' bytes:%zu :%s",
|
2019-08-21 23:57:07 -07:00
|
|
|
mxc.server,
|
|
|
|
mxc.mediaid,
|
2019-07-05 18:20:39 -07:00
|
|
|
string_view{room.room_id},
|
|
|
|
content_type,
|
|
|
|
file_size,
|
|
|
|
!permitted?
|
|
|
|
"Not permitted":
|
|
|
|
!valid_args?
|
|
|
|
"Invalid arguments":
|
|
|
|
"Unknown reason",
|
|
|
|
};
|
|
|
|
|
2020-06-12 18:06:55 -07:00
|
|
|
static const auto &addl_headers
|
|
|
|
{
|
|
|
|
"Cache-Control: public, max-age=31536000, immutable\r\n"_sv
|
|
|
|
};
|
|
|
|
|
2019-05-30 00:52:36 -07:00
|
|
|
if(fallback)
|
2019-09-28 16:12:07 -07:00
|
|
|
return m::resource::response
|
2019-05-29 17:44:10 -07:00
|
|
|
{
|
2020-06-12 18:06:55 -07:00
|
|
|
client, buf, content_type, http::OK, addl_headers
|
2019-05-29 17:44:10 -07:00
|
|
|
};
|
|
|
|
|
2019-05-30 00:52:36 -07:00
|
|
|
const auto closure{[&client, &content_type]
|
|
|
|
(const const_buffer &buf)
|
2019-05-29 17:44:10 -07:00
|
|
|
{
|
2019-09-28 16:12:07 -07:00
|
|
|
m::resource::response
|
2019-05-29 17:44:10 -07:00
|
|
|
{
|
2020-06-12 18:06:55 -07:00
|
|
|
client, buf, content_type, http::OK, addl_headers
|
2019-05-30 00:52:36 -07:00
|
|
|
};
|
|
|
|
}};
|
|
|
|
|
|
|
|
if(method == "crop")
|
|
|
|
magick::thumbcrop
|
|
|
|
{
|
|
|
|
buf, dimension, closure
|
|
|
|
};
|
|
|
|
else
|
|
|
|
magick::thumbnail
|
|
|
|
{
|
|
|
|
buf, dimension, closure
|
|
|
|
};
|
2018-04-24 18:26:35 -07:00
|
|
|
|
2019-05-29 19:20:17 -07:00
|
|
|
return {}; // responded from closure.
|
2018-02-26 04:03:07 -08:00
|
|
|
}
|