void pni_handle_transport(pn_reactor_t *reactor, pn_event_t *event) { assert(reactor); pn_transport_t *transport = pn_event_transport(event); pn_record_t *record = pn_transport_attachments(transport); pn_selectable_t *sel = (pn_selectable_t *) pn_record_get(record, PN_TRANCTX); if (sel && !pn_selectable_is_terminal(sel)) { pni_connection_update(sel); pn_reactor_update(reactor, sel); } }
const char *pn_reactor_get_connection_address(pn_reactor_t *reactor, pn_connection_t *connection) { (void)reactor; // ignored if (!connection) return NULL; pn_record_t *record = pn_connection_attachments(connection); pn_url_t *url = (pn_url_t *)pn_record_get(record, PNI_CONN_URL); if (!url) return NULL; return pn_url_str(url); }
// Get the context identified by id as a C++ T*, return null pointer if not present. template <class T> static T* ptr(id id_) { return reinterpret_cast<T*>(pn_record_get(id_.first, id_.second)); }
T* get_context(pn_record_t* record, pn_handle_t handle) { return reinterpret_cast<T*>(pn_record_get(record, handle)); }
static pn_transport_t *pni_transport(pn_selectable_t *sel) { pn_record_t *record = pn_selectable_attachments(sel); return (pn_transport_t *) pn_record_get(record, PN_TRANCTX); }
void pni_handle_bound(pn_reactor_t *reactor, pn_event_t *event) { assert(reactor); assert(event); pn_connection_t *conn = pn_event_connection(event); pn_transport_t *transport = pn_event_transport(event); pn_record_t *record = pn_connection_attachments(conn); pn_url_t *url = (pn_url_t *)pn_record_get(record, PNI_CONN_PEER_ADDRESS); const char *host = NULL; const char *port = "5672"; pn_string_t *str = NULL; // link the new transport to its reactor: pni_record_init_reactor(pn_transport_attachments(transport), reactor); if (pn_connection_acceptor(conn) != NULL) { // this connection was created by the acceptor. There is already a // socket assigned to this connection. Nothing needs to be done. return; } if (url) { host = pn_url_get_host(url); const char *uport = pn_url_get_port(url); if (uport) { port = uport; } else { const char *scheme = pn_url_get_scheme(url); if (scheme && strcmp(scheme, "amqps") == 0) { port = "5671"; } } if (!pn_connection_get_user(conn)) { // user did not manually set auth info const char *user = pn_url_get_username(url); if (user) pn_connection_set_user(conn, user); const char *passwd = pn_url_get_password(url); if (passwd) pn_connection_set_password(conn, passwd); } } else { // for backward compatibility, see if the connection's hostname can be // used for the remote address. See JIRA PROTON-1133 const char *hostname = pn_connection_get_hostname(conn); if (hostname) { str = pn_string(hostname); char *h = pn_string_buffer(str); // see if a port has been included in the hostname. This is not // allowed by the spec, but the old reactor interface allowed it. char *colon = strrchr(h, ':'); if (colon) { *colon = '\0'; port = colon + 1; } host = h; } } if (!host) { // error: no address configured pn_condition_t *cond = pn_transport_condition(transport); pn_condition_set_name(cond, "proton:io"); pn_condition_set_description(cond, "Connection failed: no address configured"); pn_transport_close_tail(transport); pn_transport_close_head(transport); } else { pn_socket_t sock = pn_connect(pni_reactor_io(reactor), host, port); // invalid sockets are ignored by poll, so we need to do this manually if (sock == PN_INVALID_SOCKET) { pn_condition_t *cond = pn_transport_condition(transport); pn_condition_set_name(cond, "proton:io"); pn_condition_set_description(cond, pn_error_text(pn_reactor_error(reactor))); pn_transport_close_tail(transport); pn_transport_close_head(transport); } else { pn_reactor_selectable_transport(reactor, sock, transport); } } pn_free(str); }