#include /** * Functions for handling the local dialer */ #include "libp2p/crypto/encoding/x509.h" #include "libp2p/conn/dialer.h" #include "libp2p/conn/connection.h" #include "libp2p/conn/transport_dialer.h" #include "libp2p/crypto/key.h" #include "libp2p/utils/linked_list.h" #include "multiaddr/multiaddr.h" #include "libp2p/net/multistream.h" #include "libp2p/secio/secio.h" #include "libp2p/yamux/yamux.h" struct TransportDialer* libp2p_conn_tcp_transport_dialer_new(); /** * Create a Dialer with the specified local information * @param peer the local peer * @param peerstore the local peerstore * @param private_key the local private key * @returns a new Dialer struct */ struct Dialer* libp2p_conn_dialer_new(struct Libp2pPeer* peer, struct Peerstore* peerstore, struct RsaPrivateKey* rsa_private_key) { int success = 0; struct Dialer* dialer = (struct Dialer*)malloc(sizeof(struct Dialer)); if (dialer != NULL) { dialer->peerstore = peerstore; dialer->peer_id = malloc(peer->id_size + 1); memset(dialer->peer_id, 0, peer->id_size + 1); if (dialer->peer_id != NULL) { strncpy(dialer->peer_id, peer->id, peer->id_size); // convert private key to rsa private key /* struct RsaPrivateKey* rsa_private_key = libp2p_crypto_rsa_rsa_private_key_new(); if (!libp2p_crypto_encoding_x509_der_to_private_key(private_key->data, private_key->data_size, rsa_private_key)) { libp2p_crypto_rsa_rsa_private_key_free(rsa_private_key); libp2p_conn_dialer_free(dialer); return NULL; } if (!libp2p_crypto_rsa_private_key_fill_public_key(rsa_private_key)) { libp2p_crypto_rsa_rsa_private_key_free(rsa_private_key); libp2p_conn_dialer_free(dialer); return NULL; } */ dialer->private_key = rsa_private_key; //TODO: build transport dialers dialer->transport_dialers = NULL; dialer->fallback_dialer = libp2p_conn_tcp_transport_dialer_new(dialer->peer_id, rsa_private_key); return dialer; } } libp2p_conn_dialer_free(dialer); return NULL; } /** * Free resources from the Dialer * NOTE: this frees the fallback dialer too (should we be doing this? * @param in the Dialer struct to free */ void libp2p_conn_dialer_free(struct Dialer* in) { if (in != NULL) { free(in->peer_id); libp2p_crypto_rsa_rsa_private_key_free(in->private_key); if (in->transport_dialers != NULL) { struct Libp2pLinkedList* current = in->transport_dialers; while(current != NULL) { libp2p_conn_transport_dialer_free((struct TransportDialer*)current->item); current = current->next; } } if (in->fallback_dialer != NULL) libp2p_conn_transport_dialer_free((struct TransportDialer*)in->fallback_dialer); free(in); } return; } /** * Retrieve a Connection struct from the dialer * NOTE: This should no longer be used. _get_stream should * be used instead (which calls this method internally). * @param dialer the dialer to use * @param muiltiaddress who to connect to * @returns a stream that is a ConnectionStream, or NULL */ struct Stream* libp2p_conn_dialer_get_connection(const struct Dialer* dialer, const struct MultiAddress* multiaddress) { struct Stream* conn = libp2p_conn_transport_dialer_get(dialer->transport_dialers, multiaddress); if (conn == NULL) { conn = dialer->fallback_dialer->dial(dialer->fallback_dialer, multiaddress); } return conn; } /*** * Attempt to connect to a particular peer. This will negotiate several protocols * @param dialer the dialer * @param peer the peer to join * @returns true(1) on success, false(0) otherwise */ int libp2p_conn_dialer_join_swarm(const struct Dialer* dialer, struct Libp2pPeer* peer, int timeout_secs) { if (dialer == NULL || peer == NULL) return 0; // find the right Multiaddress struct Libp2pLinkedList* current_entry = peer->addr_head; struct Stream* conn_stream = NULL; while (current_entry != NULL) { struct MultiAddress* ma = current_entry->item; conn_stream = libp2p_conn_dialer_get_connection(dialer, ma); if (conn_stream != NULL) { if (peer->sessionContext == NULL) { peer->sessionContext = libp2p_session_context_new(); } peer->sessionContext->insecure_stream = conn_stream; peer->sessionContext->default_stream = conn_stream; peer->sessionContext->port = multiaddress_get_ip_port(ma); multiaddress_get_ip_address(ma, &peer->sessionContext->host); break; } current_entry = current_entry->next; } if (conn_stream == NULL) return 0; // multistream struct Stream* new_stream = libp2p_net_multistream_stream_new(conn_stream); if (new_stream != NULL) { // secio over multistream new_stream = libp2p_secio_stream_new(new_stream, peer, dialer->peerstore, dialer->private_key); if (new_stream != NULL) { peer->sessionContext->default_stream = new_stream; // multistream over secio new_stream = libp2p_net_multistream_stream_new(new_stream); if (new_stream != NULL) { peer->sessionContext->default_stream = new_stream; // yamux over multistream new_stream = libp2p_yamux_stream_new(new_stream); if (new_stream != NULL) { peer->sessionContext->default_stream = new_stream; // identity over yamux // kademlia over yamux // circuit relay over yamux return 1; } } } } return 0; } /** * return a Stream that is already set up to use the passed in protocol * @param dialer the dialer to use * @param multiaddress the host to dial * @param protocol the protocol to use (right now only 'multistream' is supported) * @returns the ready-to-use stream */ struct Stream* libp2p_conn_dialer_get_stream(const struct Dialer* dialer, const struct Libp2pPeer* peer, const char* protocol) { // TODO: Implement this method return NULL; }