2017-08-29 22:03:37 +00:00
|
|
|
using Gdk;
|
|
|
|
using Gee;
|
|
|
|
|
|
|
|
using Xmpp;
|
|
|
|
using Dino.Entities;
|
|
|
|
|
|
|
|
namespace Dino {
|
|
|
|
|
|
|
|
public class FileManager : StreamInteractionModule, Object {
|
|
|
|
public static ModuleIdentity<FileManager> IDENTITY = new ModuleIdentity<FileManager>("file");
|
|
|
|
public string id { get { return IDENTITY.id; } }
|
|
|
|
|
2017-10-14 17:34:30 +00:00
|
|
|
public signal void upload_available(Account account);
|
2018-08-13 13:50:50 +00:00
|
|
|
public signal void received_file(FileTransfer file_transfer, Conversation conversation);
|
2017-08-29 22:03:37 +00:00
|
|
|
|
|
|
|
private StreamInteractor stream_interactor;
|
|
|
|
private Database db;
|
2017-10-14 17:34:30 +00:00
|
|
|
private Gee.List<FileSender> file_senders = new ArrayList<FileSender>();
|
2019-06-01 15:09:26 +00:00
|
|
|
public Gee.List<IncomingFileProcessor> incoming_processors = new ArrayList<IncomingFileProcessor>();
|
2017-10-15 22:23:51 +00:00
|
|
|
private Gee.List<OutgoingFileProcessor> outgoing_processors = new ArrayList<OutgoingFileProcessor>();
|
2017-08-29 22:03:37 +00:00
|
|
|
|
|
|
|
public static void start(StreamInteractor stream_interactor, Database db) {
|
|
|
|
FileManager m = new FileManager(stream_interactor, db);
|
|
|
|
stream_interactor.add_module(m);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static string get_storage_dir() {
|
|
|
|
return Path.build_filename(Dino.get_storage_dir(), "files");
|
|
|
|
}
|
|
|
|
|
|
|
|
private FileManager(StreamInteractor stream_interactor, Database db) {
|
|
|
|
this.stream_interactor = stream_interactor;
|
|
|
|
this.db = db;
|
|
|
|
DirUtils.create_with_parents(get_storage_dir(), 0700);
|
|
|
|
}
|
|
|
|
|
2019-02-13 20:50:15 +00:00
|
|
|
public async void send_file(string uri, Conversation conversation) {
|
2017-10-14 17:34:30 +00:00
|
|
|
FileTransfer file_transfer = new FileTransfer();
|
|
|
|
file_transfer.account = conversation.account;
|
|
|
|
file_transfer.counterpart = conversation.counterpart;
|
|
|
|
file_transfer.ourpart = conversation.account.bare_jid;
|
|
|
|
file_transfer.direction = FileTransfer.DIRECTION_SENT;
|
|
|
|
file_transfer.time = new DateTime.now_utc();
|
|
|
|
file_transfer.local_time = new DateTime.now_utc();
|
2018-11-27 13:57:52 +00:00
|
|
|
file_transfer.encryption = conversation.encryption;
|
2017-10-29 14:15:28 +00:00
|
|
|
try {
|
|
|
|
File file = File.new_for_path(uri);
|
|
|
|
FileInfo file_info = file.query_info("*", FileQueryInfoFlags.NONE);
|
|
|
|
file_transfer.file_name = file_info.get_display_name();
|
|
|
|
file_transfer.mime_type = file_info.get_content_type();
|
|
|
|
file_transfer.size = (int)file_info.get_size();
|
2019-02-13 20:50:15 +00:00
|
|
|
file_transfer.input_stream = yield file.read_async();
|
2017-10-29 14:15:28 +00:00
|
|
|
} catch (Error e) {
|
|
|
|
file_transfer.state = FileTransfer.State.FAILED;
|
|
|
|
}
|
2019-02-13 20:50:15 +00:00
|
|
|
yield save_file(file_transfer);
|
2017-10-14 17:34:30 +00:00
|
|
|
|
|
|
|
file_transfer.persist(db);
|
|
|
|
|
2017-10-15 22:23:51 +00:00
|
|
|
foreach (OutgoingFileProcessor processor in outgoing_processors) {
|
|
|
|
if (processor.can_process(conversation, file_transfer)) {
|
|
|
|
processor.process(conversation, file_transfer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-14 17:34:30 +00:00
|
|
|
foreach (FileSender file_sender in file_senders) {
|
|
|
|
if (file_sender.can_send(conversation, file_transfer)) {
|
2019-06-23 12:53:18 +00:00
|
|
|
// TODO(hrxi): Currently, this tries to send the file with every transfer available, but it should probably only select one.
|
2017-10-14 17:34:30 +00:00
|
|
|
file_sender.send_file(conversation, file_transfer);
|
2017-08-29 22:03:37 +00:00
|
|
|
}
|
2017-10-14 17:34:30 +00:00
|
|
|
}
|
2018-08-13 13:50:50 +00:00
|
|
|
received_file(file_transfer, conversation);
|
2017-10-14 17:34:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public bool is_upload_available(Conversation conversation) {
|
|
|
|
foreach (FileSender file_sender in file_senders) {
|
|
|
|
if (file_sender.is_upload_available(conversation)) return true;
|
|
|
|
}
|
|
|
|
return false;
|
2017-08-29 22:03:37 +00:00
|
|
|
}
|
|
|
|
|
2018-06-19 16:07:00 +00:00
|
|
|
public Gee.List<FileTransfer> get_latest_transfers(Account account, Jid counterpart, int n) {
|
|
|
|
Qlite.QueryBuilder select = db.file_transfer.select()
|
|
|
|
.with(db.file_transfer.counterpart_id, "=", db.get_jid_id(counterpart))
|
|
|
|
.with(db.file_transfer.account_id, "=", account.id)
|
|
|
|
.order_by(db.file_transfer.local_time, "DESC")
|
|
|
|
.limit(n);
|
2018-06-23 09:59:21 +00:00
|
|
|
return get_transfers_from_qry(select);
|
2018-06-19 16:07:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public Gee.List<FileTransfer> get_transfers_before(Account account, Jid counterpart, DateTime before, int n) {
|
|
|
|
Qlite.QueryBuilder select = db.file_transfer.select()
|
|
|
|
.with(db.file_transfer.counterpart_id, "=", db.get_jid_id(counterpart))
|
|
|
|
.with(db.file_transfer.account_id, "=", account.id)
|
|
|
|
.with(db.file_transfer.local_time, "<", (long)before.to_unix())
|
|
|
|
.order_by(db.file_transfer.local_time, "DESC")
|
|
|
|
.limit(n);
|
2018-06-23 09:59:21 +00:00
|
|
|
return get_transfers_from_qry(select);
|
2018-06-19 16:07:00 +00:00
|
|
|
}
|
|
|
|
|
2018-06-23 09:59:21 +00:00
|
|
|
public Gee.List<FileTransfer> get_transfers_after(Account account, Jid counterpart, DateTime after, int n) {
|
2017-08-29 22:03:37 +00:00
|
|
|
Qlite.QueryBuilder select = db.file_transfer.select()
|
|
|
|
.with(db.file_transfer.counterpart_id, "=", db.get_jid_id(counterpart))
|
|
|
|
.with(db.file_transfer.account_id, "=", account.id)
|
|
|
|
.with(db.file_transfer.local_time, ">", (long)after.to_unix())
|
2018-06-23 09:59:21 +00:00
|
|
|
.limit(n);
|
|
|
|
return get_transfers_from_qry(select);
|
|
|
|
}
|
2017-08-29 22:03:37 +00:00
|
|
|
|
2018-06-23 09:59:21 +00:00
|
|
|
private Gee.List<FileTransfer> get_transfers_from_qry(Qlite.QueryBuilder select) {
|
2017-08-29 22:03:37 +00:00
|
|
|
Gee.List<FileTransfer> ret = new ArrayList<FileTransfer>();
|
|
|
|
foreach (Qlite.Row row in select) {
|
2018-06-19 16:07:00 +00:00
|
|
|
FileTransfer file_transfer = new FileTransfer.from_row(db, row, get_storage_dir());
|
2017-08-29 22:03:37 +00:00
|
|
|
ret.insert(0, file_transfer);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-10-14 17:34:30 +00:00
|
|
|
public void add_provider(FileProvider file_provider) {
|
2019-06-01 15:09:26 +00:00
|
|
|
file_provider.file_incoming.connect((file_transfer, conversation) => { handle_incoming_file.begin(file_provider, file_transfer, conversation); });
|
2017-10-14 17:34:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void add_sender(FileSender file_sender) {
|
|
|
|
file_senders.add(file_sender);
|
|
|
|
file_sender.upload_available.connect((account) => {
|
|
|
|
upload_available(account);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-06-01 15:09:26 +00:00
|
|
|
public void add_incoming_processor(IncomingFileProcessor processor) {
|
|
|
|
incoming_processors.add(processor);
|
2017-10-15 22:23:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void add_outgoing_processor(OutgoingFileProcessor processor) {
|
|
|
|
outgoing_processors.add(processor);
|
|
|
|
}
|
|
|
|
|
2018-11-14 17:17:10 +00:00
|
|
|
public bool is_sender_trustworthy(FileTransfer file_transfer, Conversation conversation) {
|
|
|
|
Jid relevant_jid = stream_interactor.get_module(MucManager.IDENTITY).get_real_jid(file_transfer.from, conversation.account) ?? conversation.counterpart;
|
|
|
|
bool in_roster = stream_interactor.get_module(RosterManager.IDENTITY).get_roster_item(conversation.account, relevant_jid) != null;
|
|
|
|
return file_transfer.direction == FileTransfer.DIRECTION_SENT || in_roster;
|
|
|
|
}
|
|
|
|
|
2019-06-01 15:09:26 +00:00
|
|
|
private async void handle_incoming_file(FileProvider file_provider, FileTransfer file_transfer, Conversation conversation) {
|
2018-11-14 17:17:10 +00:00
|
|
|
if (!is_sender_trustworthy(file_transfer, conversation)) return;
|
|
|
|
|
2018-11-27 17:28:28 +00:00
|
|
|
if (file_transfer.size == -1) {
|
2018-12-30 12:35:01 +00:00
|
|
|
yield file_provider.get_meta_info(file_transfer);
|
2018-11-27 17:28:28 +00:00
|
|
|
}
|
2017-10-15 22:23:51 +00:00
|
|
|
|
2018-11-27 17:28:28 +00:00
|
|
|
if (file_transfer.size >= 0 && file_transfer.size < 5000000) {
|
|
|
|
string filename = Random.next_int().to_string("%x") + "_" + file_transfer.file_name;
|
|
|
|
File file = File.new_for_path(Path.build_filename(get_storage_dir(), filename));
|
|
|
|
yield file_provider.download(file_transfer, file);
|
2017-10-15 22:23:51 +00:00
|
|
|
|
2018-11-27 17:28:28 +00:00
|
|
|
try {
|
|
|
|
FileInfo file_info = file_transfer.get_file().query_info("*", FileQueryInfoFlags.NONE);
|
|
|
|
file_transfer.mime_type = file_info.get_content_type();
|
|
|
|
} catch (Error e) { }
|
|
|
|
|
|
|
|
file_transfer.persist(db);
|
|
|
|
received_file(file_transfer, conversation);
|
|
|
|
}
|
2017-10-15 22:23:51 +00:00
|
|
|
}
|
|
|
|
|
2019-02-13 20:50:15 +00:00
|
|
|
private async void save_file(FileTransfer file_transfer) {
|
2017-10-14 17:34:30 +00:00
|
|
|
try {
|
2017-10-29 14:15:28 +00:00
|
|
|
string filename = Random.next_int().to_string("%x") + "_" + file_transfer.file_name;
|
|
|
|
File file = File.new_for_path(Path.build_filename(get_storage_dir(), filename));
|
2017-10-14 17:34:30 +00:00
|
|
|
OutputStream os = file.create(FileCreateFlags.REPLACE_DESTINATION);
|
2019-02-13 20:50:15 +00:00
|
|
|
yield os.splice_async(file_transfer.input_stream, 0);
|
2017-10-14 17:34:30 +00:00
|
|
|
os.close();
|
|
|
|
file_transfer.state = FileTransfer.State.COMPLETE;
|
2017-10-29 14:15:28 +00:00
|
|
|
file_transfer.path = filename;
|
2019-02-13 20:50:15 +00:00
|
|
|
file_transfer.input_stream = yield file.read_async();
|
2017-10-14 17:34:30 +00:00
|
|
|
} catch (Error e) {
|
|
|
|
file_transfer.state = FileTransfer.State.FAILED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public interface FileProvider : Object {
|
2018-08-13 13:50:50 +00:00
|
|
|
public signal void file_incoming(FileTransfer file_transfer, Conversation conversation);
|
2018-11-27 17:28:28 +00:00
|
|
|
public abstract async void get_meta_info(FileTransfer file_transfer);
|
2018-11-14 17:17:10 +00:00
|
|
|
public abstract async void download(FileTransfer file_transfer, File file);
|
2017-10-14 17:34:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public interface FileSender : Object {
|
|
|
|
public signal void upload_available(Account account);
|
|
|
|
public abstract bool is_upload_available(Conversation conversation);
|
|
|
|
public abstract bool can_send(Conversation conversation, FileTransfer file_transfer);
|
|
|
|
public abstract void send_file(Conversation conversation, FileTransfer file_transfer);
|
|
|
|
}
|
|
|
|
|
2019-06-01 15:09:26 +00:00
|
|
|
public interface IncomingFileProcessor : Object {
|
2017-10-14 17:34:30 +00:00
|
|
|
public abstract bool can_process(FileTransfer file_transfer);
|
2017-10-15 22:23:51 +00:00
|
|
|
public abstract void process(FileTransfer file_transfer);
|
2017-10-14 17:34:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public interface OutgoingFileProcessor : Object {
|
2017-10-15 22:23:51 +00:00
|
|
|
public abstract bool can_process(Conversation conversation, FileTransfer file_transfer);
|
|
|
|
public abstract void process(Conversation conversation, FileTransfer file_transfer);
|
2017-08-29 22:03:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|