2017-03-02 14:37:32 +00:00
|
|
|
using Xmpp.Core;
|
|
|
|
|
|
|
|
namespace Xmpp.Tls {
|
|
|
|
private const string NS_URI = "urn:ietf:params:xml:ns:xmpp-tls";
|
|
|
|
|
|
|
|
public class Module : XmppStreamNegotiationModule {
|
2017-03-19 11:55:36 +00:00
|
|
|
public static ModuleIdentity<Module> IDENTITY = new ModuleIdentity<Module>(NS_URI, "tls_module");
|
2017-03-02 14:37:32 +00:00
|
|
|
|
|
|
|
public bool require { get; set; default = true; }
|
|
|
|
public bool server_supports_tls = false;
|
|
|
|
public bool server_requires_tls = false;
|
|
|
|
public SocketConnectable? identity = null;
|
|
|
|
|
|
|
|
public override void attach(XmppStream stream) {
|
|
|
|
stream.received_features_node.connect(this.received_features_node);
|
|
|
|
stream.received_nonza.connect(this.received_nonza);
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void detach(XmppStream stream) {
|
|
|
|
stream.received_features_node.disconnect(this.received_features_node);
|
|
|
|
stream.received_nonza.disconnect(this.received_nonza);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void received_nonza(XmppStream stream, StanzaNode node) {
|
|
|
|
if (node.ns_uri == NS_URI && node.name == "proceed") {
|
|
|
|
try {
|
2017-03-09 20:47:50 +00:00
|
|
|
var io_stream = stream.get_stream();
|
|
|
|
if (io_stream == null) return;
|
|
|
|
var conn = TlsClientConnection.new(io_stream, identity);
|
2017-03-02 14:37:32 +00:00
|
|
|
// TODO: Add certificate error handling, that is, allow the
|
|
|
|
// program to handle certificate errors. The certificate
|
|
|
|
// *is checked* by TlsClientConnection, and connection is
|
|
|
|
// not allowed to continue in case that there is an error.
|
|
|
|
stream.reset_stream(conn);
|
|
|
|
|
2017-03-19 11:55:36 +00:00
|
|
|
var flag = stream.get_flag(Flag.IDENTITY);
|
2017-03-02 14:37:32 +00:00
|
|
|
flag.peer_certificate = conn.get_peer_certificate();
|
|
|
|
flag.finished = true;
|
|
|
|
} catch (Error e) {
|
|
|
|
stderr.printf("Failed to start TLS: %s\n", e.message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void received_features_node(XmppStream stream) {
|
2017-03-19 11:55:36 +00:00
|
|
|
if (stream.has_flag(Flag.IDENTITY)) return;
|
2017-03-02 14:37:32 +00:00
|
|
|
if (stream.is_setup_needed()) return;
|
|
|
|
|
|
|
|
var starttls = stream.features.get_subnode("starttls", NS_URI);
|
|
|
|
if (starttls != null) {
|
|
|
|
server_supports_tls = true;
|
|
|
|
if (starttls.get_subnode("required") != null || stream.features.get_all_subnodes().size == 1) {
|
|
|
|
server_requires_tls = true;
|
|
|
|
}
|
|
|
|
if (server_requires_tls || require) {
|
|
|
|
try {
|
|
|
|
stream.write(new StanzaNode.build("starttls", NS_URI).add_self_xmlns());
|
|
|
|
} catch (IOStreamError e) {
|
|
|
|
stderr.printf("Failed to request TLS: %s\n", e.message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (identity == null) {
|
|
|
|
identity = new NetworkService("xmpp-client", "tcp", stream.remote_name);
|
|
|
|
}
|
|
|
|
stream.add_flag(new Flag());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public override bool mandatory_outstanding(XmppStream stream) {
|
2017-03-19 11:55:36 +00:00
|
|
|
return require && (!stream.has_flag(Flag.IDENTITY) || !stream.get_flag(Flag.IDENTITY).finished);
|
2017-03-02 14:37:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public override bool negotiation_active(XmppStream stream) {
|
2017-03-19 11:55:36 +00:00
|
|
|
return stream.has_flag(Flag.IDENTITY) && !stream.get_flag(Flag.IDENTITY).finished;
|
2017-03-02 14:37:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public override string get_ns() { return NS_URI; }
|
2017-03-19 11:55:36 +00:00
|
|
|
public override string get_id() { return IDENTITY.id; }
|
2017-03-02 14:37:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public class Flag : XmppStreamFlag {
|
2017-03-19 11:55:36 +00:00
|
|
|
public static FlagIdentity<Flag> IDENTITY = new FlagIdentity<Flag>(NS_URI, "tls");
|
2017-03-02 14:37:32 +00:00
|
|
|
public TlsCertificate? peer_certificate;
|
2017-08-09 18:44:15 +00:00
|
|
|
public bool finished { get; set; default=false; }
|
2017-03-02 14:37:32 +00:00
|
|
|
|
|
|
|
public override string get_ns() { return NS_URI; }
|
2017-03-19 11:55:36 +00:00
|
|
|
public override string get_id() { return IDENTITY.id; }
|
2017-03-02 14:37:32 +00:00
|
|
|
}
|
|
|
|
}
|