anotherim-desktop/libdino/src/service/stream_interactor.vala
hrxi 148cf48d2b Add libnice and listen for direct connections in Jingle SOCKS5 (#608)
Add libnice as a plugin. If it is present, use libnice to enumerate
local IP addresses and listen on them to support direct connections for
Jingle SOCKS5.

Tested with Conversations and Gajim.

Created the nice.vapi file using
```
vapigen --library nice --pkg gio-2.0 --metadatadir metadata /usr/share/gir-1.0/Nice-0.1.gir
```
2021-03-19 22:06:02 +01:00

105 lines
3.2 KiB
Vala

using Gee;
using Xmpp;
using Dino.Entities;
namespace Dino {
public class StreamInteractor : Object {
public signal void account_added(Account account);
public signal void account_removed(Account account);
public signal void stream_resumed(Account account, XmppStream stream);
public signal void stream_negotiated(Account account, XmppStream stream);
public signal void stream_attached_modules(Account account, XmppStream stream);
public ModuleManager module_manager;
public ConnectionManager connection_manager;
private ArrayList<StreamInteractionModule> modules = new ArrayList<StreamInteractionModule>();
public StreamInteractor(Database db) {
module_manager = new ModuleManager();
connection_manager = new ConnectionManager(module_manager);
connection_manager.stream_opened.connect(on_stream_opened);
connection_manager.stream_attached_modules.connect((account, stream) => {
stream_attached_modules(account, stream);
});
}
public void connect_account(Account account) {
module_manager.initialize(account);
account_added(account);
connection_manager.connect_account(account);
}
public async void disconnect_account(Account account) {
yield connection_manager.disconnect_account(account);
account_removed(account);
}
public ArrayList<Account> get_accounts() {
ArrayList<Account> ret = new ArrayList<Account>(Account.equals_func);
foreach (Account account in connection_manager.get_managed_accounts()) {
ret.add(account);
}
return ret;
}
public XmppStream? get_stream(Account account) {
return connection_manager.get_stream(account);
}
public void add_module(StreamInteractionModule module) {
modules.add(module);
}
public T? get_module<T>(ModuleIdentity<T>? identity) {
if (identity == null) return null;
foreach (StreamInteractionModule module in modules) {
if (identity.matches(module)) return identity.cast(module);
}
return null;
}
public new T? get<T>() {
foreach (StreamInteractionModule module in modules) {
if (module.get_type() == typeof(T)) return (T?) module;
}
return null;
}
private void on_stream_opened(Account account, XmppStream stream) {
stream.stream_negotiated.connect( (stream) => {
var flag = stream.get_flag(Xep.StreamManagement.Flag.IDENTITY);
if (flag == null || flag.resumed == false) {
stream_negotiated(account, stream);
} else if (flag != null && flag.resumed == true) {
stream_resumed(account, stream);
}
});
}
}
public class ModuleIdentity<T> : Object {
public string id { get; private set; }
public ModuleIdentity(string id) {
this.id = id;
}
public T? cast(StreamInteractionModule module) {
return module.get_type().is_a(typeof(T)) ? (T?) module : null;
}
public bool matches(StreamInteractionModule module) {
return module.id == id;
}
}
public interface StreamInteractionModule : Object {
public abstract string id { get; }
}
}