2018-02-26 13:03:07 +01: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-02 11:08:06 +01:00
|
|
|
#include <boost/gil/image.hpp>
|
|
|
|
#include <boost/gil/typedefs.hpp>
|
|
|
|
#include <boost/gil/extension/io/jpeg_io.hpp>
|
|
|
|
// #include <boost/gil/extension/numeric/sampler.hpp>
|
|
|
|
// #include <boost/gil/extension/numeric/resample.hpp>
|
|
|
|
|
2018-03-08 21:38:02 +01:00
|
|
|
#include "media.h"
|
2018-02-26 13:03:07 +01:00
|
|
|
|
|
|
|
resource
|
|
|
|
thumbnail_resource__legacy
|
|
|
|
{
|
|
|
|
"/_matrix/media/v1/thumbnail/",
|
|
|
|
{
|
2018-04-12 23:44:30 +02:00
|
|
|
"(11.7.1.4) thumbnails (legacy version)",
|
2018-02-26 13:03:07 +01:00
|
|
|
resource::DIRECTORY,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
resource
|
|
|
|
thumbnail_resource
|
|
|
|
{
|
|
|
|
"/_matrix/media/r0/thumbnail/",
|
|
|
|
{
|
2018-04-12 23:44:30 +02:00
|
|
|
"(11.7.1.4) thumbnails",
|
2018-02-26 13:03:07 +01:00
|
|
|
resource::DIRECTORY,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static resource::response
|
|
|
|
get__thumbnail_remote(client &client,
|
|
|
|
const resource::request &request,
|
2018-04-23 05:46:14 +02:00
|
|
|
const string_view &remote,
|
2018-02-26 13:03:07 +01:00
|
|
|
const string_view &server,
|
2018-03-08 21:38:02 +01:00
|
|
|
const string_view &file,
|
|
|
|
const m::room &room);
|
|
|
|
|
|
|
|
static resource::response
|
|
|
|
get__thumbnail_local(client &client,
|
|
|
|
const resource::request &request,
|
|
|
|
const string_view &server,
|
|
|
|
const string_view &file,
|
|
|
|
const m::room &room);
|
2018-02-26 13:03:07 +01:00
|
|
|
|
|
|
|
resource::response
|
|
|
|
get__thumbnail(client &client,
|
|
|
|
const resource::request &request)
|
|
|
|
{
|
|
|
|
if(request.parv.size() < 1)
|
|
|
|
throw http::error
|
|
|
|
{
|
|
|
|
http::MULTIPLE_CHOICES, "Server name parameter required"
|
|
|
|
};
|
|
|
|
|
|
|
|
if(request.parv.size() < 2)
|
|
|
|
throw http::error
|
|
|
|
{
|
|
|
|
http::MULTIPLE_CHOICES, "Media ID parameter required"
|
|
|
|
};
|
|
|
|
|
2018-04-23 05:46:14 +02:00
|
|
|
auto &server
|
2018-02-26 13:03:07 +01:00
|
|
|
{
|
|
|
|
request.parv[0]
|
|
|
|
};
|
|
|
|
|
|
|
|
const auto &file
|
|
|
|
{
|
|
|
|
request.parv[1]
|
|
|
|
};
|
|
|
|
|
2018-03-08 21:38:02 +01:00
|
|
|
const m::room::id::buf room_id
|
2018-02-26 13:03:07 +01:00
|
|
|
{
|
2018-03-08 21:38:02 +01:00
|
|
|
file_room_id(server, file)
|
2018-02-26 13:03:07 +01:00
|
|
|
};
|
|
|
|
|
2018-04-06 08:07:55 +02:00
|
|
|
m::vm::opts::commit vmopts;
|
2018-04-13 00:58:30 +02:00
|
|
|
vmopts.history = false;
|
2018-03-08 21:38:02 +01:00
|
|
|
const m::room room
|
2018-02-26 13:03:07 +01:00
|
|
|
{
|
2018-04-13 00:58:30 +02:00
|
|
|
room_id, &vmopts
|
2018-02-26 13:03:07 +01:00
|
|
|
};
|
|
|
|
|
2018-04-13 00:58:30 +02:00
|
|
|
//TODO: ABA
|
2018-03-08 21:38:02 +01:00
|
|
|
if(m::exists(room))
|
|
|
|
return get__thumbnail_local(client, request, server, file, room);
|
|
|
|
|
2018-04-23 05:46:14 +02:00
|
|
|
//TODO: XXX conf
|
|
|
|
//TODO: XXX vector
|
|
|
|
static const string_view secondary
|
|
|
|
{
|
|
|
|
"matrix.org"
|
|
|
|
};
|
|
|
|
|
|
|
|
const string_view &remote
|
|
|
|
{
|
|
|
|
my_host(server)?
|
|
|
|
secondary:
|
|
|
|
server
|
|
|
|
};
|
|
|
|
|
|
|
|
if(!my_host(remote))
|
2018-04-13 00:58:30 +02:00
|
|
|
{
|
|
|
|
//TODO: ABA TXN
|
|
|
|
create(room, m::me.user_id, "file");
|
2018-04-23 05:46:14 +02:00
|
|
|
return get__thumbnail_remote(client, request, remote, server, file, room);
|
2018-04-13 00:58:30 +02:00
|
|
|
}
|
2018-03-08 21:38:02 +01:00
|
|
|
|
|
|
|
throw m::NOT_FOUND
|
2018-02-26 13:03:07 +01:00
|
|
|
{
|
2018-03-08 21:38:02 +01:00
|
|
|
"Media not found"
|
2018-02-26 13:03:07 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-04-12 23:44:30 +02:00
|
|
|
static resource::method
|
2018-02-26 13:03:07 +01:00
|
|
|
method_get__legacy
|
|
|
|
{
|
|
|
|
thumbnail_resource__legacy, "GET", get__thumbnail
|
|
|
|
};
|
|
|
|
|
2018-04-12 23:44:30 +02:00
|
|
|
static resource::method
|
2018-02-26 13:03:07 +01:00
|
|
|
method_get
|
|
|
|
{
|
|
|
|
thumbnail_resource, "GET", get__thumbnail
|
|
|
|
};
|
|
|
|
|
|
|
|
static resource::response
|
|
|
|
get__thumbnail_remote(client &client,
|
|
|
|
const resource::request &request,
|
|
|
|
const string_view &hostname,
|
2018-04-23 05:46:14 +02:00
|
|
|
const string_view &server,
|
2018-03-08 21:38:02 +01:00
|
|
|
const string_view &mediaid,
|
|
|
|
const m::room &room)
|
2018-04-13 00:58:30 +02:00
|
|
|
try
|
2018-02-26 13:03:07 +01:00
|
|
|
{
|
|
|
|
const net::hostport remote
|
|
|
|
{
|
|
|
|
hostname
|
|
|
|
};
|
|
|
|
|
2018-04-23 05:44:15 +02:00
|
|
|
const unique_buffer<mutable_buffer> buf
|
|
|
|
{
|
|
|
|
16_KiB
|
|
|
|
};
|
|
|
|
|
2018-02-27 10:42:25 +01:00
|
|
|
window_buffer wb{buf};
|
2018-04-23 05:44:15 +02:00
|
|
|
thread_local char uri[4_KiB];
|
2018-02-26 13:03:07 +01:00
|
|
|
http::request
|
|
|
|
{
|
|
|
|
wb, hostname, "GET", fmt::sprintf
|
|
|
|
{
|
2018-04-23 05:46:14 +02:00
|
|
|
uri, "/_matrix/media/r0/download/%s/%s", server, mediaid
|
2018-02-26 13:03:07 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-27 10:42:25 +01:00
|
|
|
const const_buffer out_head
|
|
|
|
{
|
|
|
|
wb.completed()
|
|
|
|
};
|
|
|
|
|
|
|
|
// Remaining space in buffer is used for received head
|
|
|
|
const mutable_buffer in_head
|
|
|
|
{
|
2018-04-23 05:44:15 +02:00
|
|
|
data(buf) + size(out_head), size(buf) - size(out_head)
|
2018-02-27 10:42:25 +01:00
|
|
|
};
|
2018-02-26 13:03:07 +01:00
|
|
|
|
2018-03-08 21:38:02 +01:00
|
|
|
//TODO: --- This should use the progress callback to build blocks
|
|
|
|
|
2018-02-27 10:42:25 +01:00
|
|
|
// Null content buffer will cause dynamic allocation internally.
|
|
|
|
const mutable_buffer in_content{};
|
|
|
|
|
|
|
|
struct server::request::opts opts;
|
2018-02-26 13:03:07 +01:00
|
|
|
server::request remote_request
|
|
|
|
{
|
2018-02-27 10:42:25 +01:00
|
|
|
remote, { out_head }, { in_head, in_content }, &opts
|
2018-02-26 13:03:07 +01:00
|
|
|
};
|
|
|
|
|
2018-04-06 06:19:16 +02:00
|
|
|
if(!remote_request.wait(seconds(10), std::nothrow))
|
2018-02-26 13:03:07 +01:00
|
|
|
throw http::error
|
|
|
|
{
|
|
|
|
http::REQUEST_TIMEOUT
|
|
|
|
};
|
|
|
|
|
2018-03-08 21:38:02 +01:00
|
|
|
//TODO: ---
|
|
|
|
|
2018-02-26 13:03:07 +01:00
|
|
|
const auto &code
|
|
|
|
{
|
|
|
|
remote_request.get()
|
|
|
|
};
|
|
|
|
|
|
|
|
char mime_type_buf[64];
|
|
|
|
const string_view content_type
|
|
|
|
{
|
|
|
|
magic::mime(mime_type_buf, remote_request.in.content)
|
|
|
|
};
|
|
|
|
|
2018-03-08 21:38:02 +01:00
|
|
|
const size_t file_size
|
|
|
|
{
|
|
|
|
size(remote_request.in.content)
|
|
|
|
};
|
|
|
|
|
2018-02-26 13:03:07 +01:00
|
|
|
parse::buffer pb{remote_request.in.head};
|
|
|
|
parse::capstan pc{pb};
|
|
|
|
pc.read += size(remote_request.in.head);
|
|
|
|
const http::response::head head{pc};
|
|
|
|
if(content_type != head.content_type)
|
|
|
|
log::warning
|
|
|
|
{
|
|
|
|
"Server %s claims thumbnail %s is '%s' but we think it is '%s'",
|
|
|
|
hostname,
|
|
|
|
mediaid,
|
|
|
|
head.content_type,
|
|
|
|
content_type
|
|
|
|
};
|
|
|
|
|
2018-04-13 00:58:30 +02:00
|
|
|
const size_t written
|
2018-03-08 21:38:02 +01:00
|
|
|
{
|
2018-04-13 00:58:30 +02:00
|
|
|
write_file(room, remote_request.in.content, content_type)
|
2018-03-08 21:38:02 +01:00
|
|
|
};
|
|
|
|
|
2018-04-25 03:26:35 +02:00
|
|
|
return resource::response
|
|
|
|
{
|
|
|
|
client, remote_request.in.content, content_type
|
|
|
|
};
|
2018-03-08 21:38:02 +01:00
|
|
|
}
|
2018-04-13 00:58:30 +02:00
|
|
|
catch(const ircd::server::unavailable &e)
|
|
|
|
{
|
|
|
|
throw http::error
|
|
|
|
{
|
|
|
|
http::BAD_GATEWAY, e.what()
|
|
|
|
};
|
|
|
|
}
|
2018-03-08 21:38:02 +01:00
|
|
|
|
|
|
|
static resource::response
|
|
|
|
get__thumbnail_local(client &client,
|
|
|
|
const resource::request &request,
|
|
|
|
const string_view &hostname,
|
|
|
|
const string_view &mediaid,
|
|
|
|
const m::room &room)
|
|
|
|
{
|
|
|
|
// Get the file's total size
|
|
|
|
size_t file_size{0};
|
|
|
|
room.get("ircd.file.stat", "size", [&file_size]
|
|
|
|
(const m::event &event)
|
|
|
|
{
|
|
|
|
file_size = at<"content"_>(event).get<size_t>("value");
|
|
|
|
});
|
|
|
|
|
|
|
|
// Get the MIME type
|
|
|
|
char type_buf[64];
|
2018-04-13 00:58:30 +02:00
|
|
|
string_view content_type
|
|
|
|
{
|
|
|
|
"application/octet-stream"
|
|
|
|
};
|
|
|
|
|
2018-03-08 21:38:02 +01:00
|
|
|
room.get("ircd.file.stat", "type", [&type_buf, &content_type]
|
|
|
|
(const m::event &event)
|
|
|
|
{
|
|
|
|
const auto &value
|
|
|
|
{
|
|
|
|
unquote(at<"content"_>(event).at("value"))
|
|
|
|
};
|
|
|
|
|
|
|
|
content_type =
|
|
|
|
{
|
|
|
|
type_buf, copy(type_buf, value)
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
// Send HTTP head to client
|
2018-04-23 05:44:15 +02:00
|
|
|
const resource::response response
|
2018-03-08 21:38:02 +01:00
|
|
|
{
|
2018-03-09 22:15:32 +01:00
|
|
|
client, http::OK, content_type, file_size
|
2018-03-08 21:38:02 +01:00
|
|
|
};
|
|
|
|
|
2018-04-23 05:44:15 +02:00
|
|
|
size_t sent{0};
|
|
|
|
const auto sink{[&client, &sent]
|
2018-04-25 03:26:35 +02:00
|
|
|
(const const_buffer &block)
|
2018-03-08 21:38:02 +01:00
|
|
|
{
|
2018-04-23 05:44:15 +02:00
|
|
|
sent += client.write_all(block);
|
|
|
|
}};
|
|
|
|
|
|
|
|
const size_t read
|
|
|
|
{
|
|
|
|
read_each_block(room, sink)
|
|
|
|
};
|
2018-03-08 21:38:02 +01:00
|
|
|
|
2018-04-25 03:26:35 +02:00
|
|
|
if(unlikely(read != file_size)) log::error
|
|
|
|
{
|
|
|
|
media_log, "File %s/%s [%s] size mismatch: expected %zu got %zu",
|
|
|
|
hostname,
|
|
|
|
mediaid,
|
|
|
|
string_view{room.room_id},
|
|
|
|
file_size,
|
|
|
|
read
|
|
|
|
};
|
|
|
|
|
|
|
|
// Have to kill client here after failing content length expectation.
|
|
|
|
if(unlikely(read != file_size))
|
|
|
|
client.close(net::dc::RST, net::close_ignore);
|
|
|
|
|
2018-04-23 05:44:15 +02:00
|
|
|
assert(read == sent);
|
|
|
|
return response;
|
2018-02-26 13:03:07 +01:00
|
|
|
}
|