mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-12 15:36:17 +01:00
Add support for MX, TXT, NS and SOA records to GResolver
* Add resolver functions for looking up DNS records of various types. Currently implemented: MX, TXT, SOA, SRV, NS * Return records as GVariant tuples. * Make the GSrvTarget lookups a wrapper over this new functionality. * Rework the resolver test so that it has support for looking up MX, NS, SOA, TXT records, and uses GOptionContext https://bugzilla.gnome.org/show_bug.cgi?id=672944
This commit is contained in:
parent
cec17df598
commit
666374c16f
@ -1699,6 +1699,9 @@ g_resolver_lookup_service
|
|||||||
g_resolver_lookup_service_async
|
g_resolver_lookup_service_async
|
||||||
g_resolver_lookup_service_finish
|
g_resolver_lookup_service_finish
|
||||||
g_resolver_free_targets
|
g_resolver_free_targets
|
||||||
|
g_resolver_lookup_records
|
||||||
|
g_resolver_lookup_records_async
|
||||||
|
g_resolver_lookup_records_finish
|
||||||
|
|
||||||
<SUBSECTION>
|
<SUBSECTION>
|
||||||
G_RESOLVER_ERROR
|
G_RESOLVER_ERROR
|
||||||
@ -1717,6 +1720,7 @@ G_TYPE_RESOLVER
|
|||||||
GResolverPrivate
|
GResolverPrivate
|
||||||
g_resolver_get_type
|
g_resolver_get_type
|
||||||
g_resolver_error_quark
|
g_resolver_error_quark
|
||||||
|
g_resolver_record_type_get_type
|
||||||
</SECTION>
|
</SECTION>
|
||||||
|
|
||||||
<SECTION>
|
<SECTION>
|
||||||
|
@ -1713,3 +1713,7 @@ g_resources_unregister
|
|||||||
g_static_resource_fini
|
g_static_resource_fini
|
||||||
g_static_resource_get_resource
|
g_static_resource_get_resource
|
||||||
g_static_resource_init
|
g_static_resource_init
|
||||||
|
g_resolver_lookup_records
|
||||||
|
g_resolver_lookup_records_async
|
||||||
|
g_resolver_lookup_records_finish
|
||||||
|
g_resolver_record_type_get_type
|
||||||
|
@ -630,6 +630,49 @@ typedef enum {
|
|||||||
G_RESOLVER_ERROR_INTERNAL
|
G_RESOLVER_ERROR_INTERNAL
|
||||||
} GResolverError;
|
} GResolverError;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GResolverRecordType:
|
||||||
|
* @G_RESOLVER_RECORD_SRV: lookup DNS SRV records for a domain
|
||||||
|
* @G_RESOLVER_RECORD_MX: lookup DNS MX records for a domain
|
||||||
|
* @G_RESOLVER_RECORD_TXT: lookup DNS TXT records for a name
|
||||||
|
* @G_RESOLVER_RECORD_SOA: lookup DNS SOA records for a zone
|
||||||
|
* @G_RESOLVER_RECORD_NS: lookup DNS NS records for a domain
|
||||||
|
*
|
||||||
|
* The type of record that g_resolver_lookup_records() or
|
||||||
|
* g_resolver_lookup_records_async() should retrieve. The records are returned
|
||||||
|
* as lists of #GVariant tuples. Each record type has different values in
|
||||||
|
* the variant tuples returned.
|
||||||
|
*
|
||||||
|
* %G_RESOLVER_RECORD_SRV records are returned as variants with the signature
|
||||||
|
* '(qqqs)', containing a guint16 with the priority, a guint16 with the
|
||||||
|
* weight, a guint16 with the port, and a string of the hostname.
|
||||||
|
*
|
||||||
|
* %G_RESOLVER_RECORD_MX records are returned as variants with the signature
|
||||||
|
* '(qs)', representing a guint16 with the preference, and a string containing
|
||||||
|
* the mail exchanger hostname.
|
||||||
|
*
|
||||||
|
* %G_RESOLVER_RECORD_TXT records are returned as variants with the signature
|
||||||
|
* '(as)', representing an array of the strings in the text record.
|
||||||
|
*
|
||||||
|
* %G_RESOLVER_RECORD_SOA records are returned as variants with the signature
|
||||||
|
* '(ssuuuuu)', representing a string containing the primary name server, a
|
||||||
|
* string containing the administrator, the serial as a guint32, the refresh
|
||||||
|
* interval as guint32, the retry interval as a guint32, the expire timeout
|
||||||
|
* as a guint32, and the ttl as a guint32.
|
||||||
|
*
|
||||||
|
* %G_RESOLVER_RECORD_NS records are returned as variants with the signature
|
||||||
|
* '(s)', representing a string of the hostname of the name server.
|
||||||
|
*
|
||||||
|
* Since: 2.34
|
||||||
|
*/
|
||||||
|
typedef enum {
|
||||||
|
G_RESOLVER_RECORD_SRV = 1,
|
||||||
|
G_RESOLVER_RECORD_MX,
|
||||||
|
G_RESOLVER_RECORD_TXT,
|
||||||
|
G_RESOLVER_RECORD_SOA,
|
||||||
|
G_RESOLVER_RECORD_NS
|
||||||
|
} GResolverRecordType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* GResourceError:
|
* GResourceError:
|
||||||
* @G_RESOURCE_ERROR_NOT_FOUND: no file was found at the requested path
|
* @G_RESOURCE_ERROR_NOT_FOUND: no file was found at the requested path
|
||||||
|
@ -95,13 +95,19 @@ char *_g_resolver_name_from_nameinfo (GInetAddress *address,
|
|||||||
GError **error);
|
GError **error);
|
||||||
|
|
||||||
#if defined(G_OS_UNIX)
|
#if defined(G_OS_UNIX)
|
||||||
GList *_g_resolver_targets_from_res_query (const gchar *rrname,
|
gint _g_resolver_record_type_to_rrtype (GResolverRecordType record_type);
|
||||||
|
|
||||||
|
GList *_g_resolver_records_from_res_query (const gchar *rrname,
|
||||||
|
gint rrtype,
|
||||||
guchar *answer,
|
guchar *answer,
|
||||||
gint len,
|
gint len,
|
||||||
gint herr,
|
gint herr,
|
||||||
GError **error);
|
GError **error);
|
||||||
#elif defined(G_OS_WIN32)
|
#elif defined(G_OS_WIN32)
|
||||||
GList *_g_resolver_targets_from_DnsQuery (const gchar *rrname,
|
WORD _g_resolver_record_type_to_dnstype (GResolverRecordType record_type);
|
||||||
|
|
||||||
|
GList *_g_resolver_records_from_DnsQuery (const gchar *rrname,
|
||||||
|
WORD dnstype,
|
||||||
DNS_STATUS status,
|
DNS_STATUS status,
|
||||||
DNS_RECORD *results,
|
DNS_RECORD *results,
|
||||||
GError **error);
|
GError **error);
|
||||||
|
471
gio/gresolver.c
471
gio/gresolver.c
@ -78,11 +78,81 @@ struct _GResolverPrivate {
|
|||||||
*/
|
*/
|
||||||
G_DEFINE_TYPE (GResolver, g_resolver, G_TYPE_OBJECT)
|
G_DEFINE_TYPE (GResolver, g_resolver, G_TYPE_OBJECT)
|
||||||
|
|
||||||
|
static GList *
|
||||||
|
srv_records_to_targets (GList *records)
|
||||||
|
{
|
||||||
|
const gchar *hostname;
|
||||||
|
guint16 port, priority, weight;
|
||||||
|
GSrvTarget *target;
|
||||||
|
GList *l;
|
||||||
|
|
||||||
|
for (l = records; l != NULL; l = g_list_next (l))
|
||||||
|
{
|
||||||
|
g_variant_get (l->data, "(qqq&s)", &priority, &weight, &port, &hostname);
|
||||||
|
target = g_srv_target_new (hostname, port, priority, weight);
|
||||||
|
g_variant_unref (l->data);
|
||||||
|
l->data = target;
|
||||||
|
}
|
||||||
|
|
||||||
|
return g_srv_target_list_sort (records);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GList *
|
||||||
|
g_resolver_real_lookup_service (GResolver *resolver,
|
||||||
|
const gchar *rrname,
|
||||||
|
GCancellable *cancellable,
|
||||||
|
GError **error)
|
||||||
|
{
|
||||||
|
GList *records;
|
||||||
|
|
||||||
|
records = G_RESOLVER_GET_CLASS (resolver)->lookup_records (resolver,
|
||||||
|
rrname,
|
||||||
|
G_RESOLVER_RECORD_SRV,
|
||||||
|
cancellable,
|
||||||
|
error);
|
||||||
|
|
||||||
|
return srv_records_to_targets (records);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
g_resolver_real_lookup_service_async (GResolver *resolver,
|
||||||
|
const gchar *rrname,
|
||||||
|
GCancellable *cancellable,
|
||||||
|
GAsyncReadyCallback callback,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
G_RESOLVER_GET_CLASS (resolver)->lookup_records_async (resolver,
|
||||||
|
rrname,
|
||||||
|
G_RESOLVER_RECORD_SRV,
|
||||||
|
cancellable,
|
||||||
|
callback,
|
||||||
|
user_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GList *
|
||||||
|
g_resolver_real_lookup_service_finish (GResolver *resolver,
|
||||||
|
GAsyncResult *result,
|
||||||
|
GError **error)
|
||||||
|
{
|
||||||
|
GList *records;
|
||||||
|
|
||||||
|
records = G_RESOLVER_GET_CLASS (resolver)->lookup_records_finish (resolver,
|
||||||
|
result,
|
||||||
|
error);
|
||||||
|
|
||||||
|
return srv_records_to_targets (records);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
g_resolver_class_init (GResolverClass *resolver_class)
|
g_resolver_class_init (GResolverClass *resolver_class)
|
||||||
{
|
{
|
||||||
volatile GType type;
|
volatile GType type;
|
||||||
|
|
||||||
|
/* Automatically pass these over to the lookup_records methods */
|
||||||
|
resolver_class->lookup_service = g_resolver_real_lookup_service;
|
||||||
|
resolver_class->lookup_service_async = g_resolver_real_lookup_service_async;
|
||||||
|
resolver_class->lookup_service_finish = g_resolver_real_lookup_service_finish;
|
||||||
|
|
||||||
g_type_class_add_private (resolver_class, sizeof (GResolverPrivate));
|
g_type_class_add_private (resolver_class, sizeof (GResolverPrivate));
|
||||||
|
|
||||||
/* Make sure _g_networking_init() has been called */
|
/* Make sure _g_networking_init() has been called */
|
||||||
@ -707,6 +777,112 @@ g_resolver_free_targets (GList *targets)
|
|||||||
g_list_free (targets);
|
g_list_free (targets);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* g_resolver_lookup_records:
|
||||||
|
* @resolver: a #GResolver
|
||||||
|
* @rrname: the DNS name to lookup the record for
|
||||||
|
* @record_type: the type of DNS record to lookup
|
||||||
|
* @cancellable: (allow-none): a #GCancellable, or %NULL
|
||||||
|
* @error: return location for a #GError, or %NULL
|
||||||
|
*
|
||||||
|
* Synchronously performs a DNS record lookup for the given @rrname and returns
|
||||||
|
* a list of records as #GVariant tuples. See #GResolverRecordType for
|
||||||
|
* information on what the records contain for each @record_type.
|
||||||
|
*
|
||||||
|
* If the DNS resolution fails, @error (if non-%NULL) will be set to
|
||||||
|
* a value from #GResolverError.
|
||||||
|
*
|
||||||
|
* If @cancellable is non-%NULL, it can be used to cancel the
|
||||||
|
* operation, in which case @error (if non-%NULL) will be set to
|
||||||
|
* %G_IO_ERROR_CANCELLED.
|
||||||
|
*
|
||||||
|
* Return value: (element-type GVariant) (transfer full): a #GList of #GVariant,
|
||||||
|
* or %NULL on error. You must free each of the records and the list when you are
|
||||||
|
* done with it. (You can use g_list_free_full() with g_variant_unref() to do this.)
|
||||||
|
*
|
||||||
|
* Since: 2.34
|
||||||
|
*/
|
||||||
|
GList *
|
||||||
|
g_resolver_lookup_records (GResolver *resolver,
|
||||||
|
const gchar *rrname,
|
||||||
|
GResolverRecordType record_type,
|
||||||
|
GCancellable *cancellable,
|
||||||
|
GError **error)
|
||||||
|
{
|
||||||
|
GList *records;
|
||||||
|
|
||||||
|
g_return_val_if_fail (G_IS_RESOLVER (resolver), NULL);
|
||||||
|
g_return_val_if_fail (rrname != NULL, NULL);
|
||||||
|
|
||||||
|
g_resolver_maybe_reload (resolver);
|
||||||
|
records = G_RESOLVER_GET_CLASS (resolver)->
|
||||||
|
lookup_records (resolver, rrname, record_type, cancellable, error);
|
||||||
|
|
||||||
|
return records;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* g_resolver_lookup_records_async:
|
||||||
|
* @resolver: a #GResolver
|
||||||
|
* @rrname: the DNS name to lookup the record for
|
||||||
|
* @record_type: the type of DNS record to lookup
|
||||||
|
* @cancellable: (allow-none): a #GCancellable, or %NULL
|
||||||
|
* @callback: (scope async): callback to call after resolution completes
|
||||||
|
* @user_data: (closure): data for @callback
|
||||||
|
*
|
||||||
|
* Begins asynchronously performing a DNS lookup for the given
|
||||||
|
* @rrname, and eventually calls @callback, which must call
|
||||||
|
* g_resolver_lookup_records_finish() to get the final result. See
|
||||||
|
* g_resolver_lookup_records() for more details.
|
||||||
|
*
|
||||||
|
* Since: 2.34
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
g_resolver_lookup_records_async (GResolver *resolver,
|
||||||
|
const gchar *rrname,
|
||||||
|
GResolverRecordType record_type,
|
||||||
|
GCancellable *cancellable,
|
||||||
|
GAsyncReadyCallback callback,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
g_return_if_fail (G_IS_RESOLVER (resolver));
|
||||||
|
g_return_if_fail (rrname != NULL);
|
||||||
|
|
||||||
|
g_resolver_maybe_reload (resolver);
|
||||||
|
G_RESOLVER_GET_CLASS (resolver)->
|
||||||
|
lookup_records_async (resolver, rrname, record_type, cancellable, callback, user_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* g_resolver_lookup_records_finish:
|
||||||
|
* @resolver: a #GResolver
|
||||||
|
* @result: the result passed to your #GAsyncReadyCallback
|
||||||
|
* @error: return location for a #GError, or %NULL
|
||||||
|
*
|
||||||
|
* Retrieves the result of a previous call to
|
||||||
|
* g_resolver_lookup_records_async(). Returns a list of records as #GVariant
|
||||||
|
* tuples. See #GResolverRecordType for information on what the records contain.
|
||||||
|
*
|
||||||
|
* If the DNS resolution failed, @error (if non-%NULL) will be set to
|
||||||
|
* a value from #GResolverError. If the operation was cancelled,
|
||||||
|
* @error will be set to %G_IO_ERROR_CANCELLED.
|
||||||
|
*
|
||||||
|
* Return value: (element-type GVariant) (transfer full): a #GList of #GVariant,
|
||||||
|
* or %NULL on error. You must free each of the records and the list when you are
|
||||||
|
* done with it. (You can use g_list_free_full() with g_variant_unref() to do this.)
|
||||||
|
*
|
||||||
|
* Since: 2.34
|
||||||
|
*/
|
||||||
|
GList *
|
||||||
|
g_resolver_lookup_records_finish (GResolver *resolver,
|
||||||
|
GAsyncResult *result,
|
||||||
|
GError **error)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (G_IS_RESOLVER (resolver), NULL);
|
||||||
|
return G_RESOLVER_GET_CLASS (resolver)->
|
||||||
|
lookup_records_finish (resolver, result, error);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* g_resolver_error_quark:
|
* g_resolver_error_quark:
|
||||||
*
|
*
|
||||||
@ -821,9 +997,133 @@ _g_resolver_name_from_nameinfo (GInetAddress *address,
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if defined(G_OS_UNIX)
|
#if defined(G_OS_UNIX)
|
||||||
|
static GVariant *
|
||||||
|
parse_res_srv (guchar *answer,
|
||||||
|
guchar *end,
|
||||||
|
guchar **p)
|
||||||
|
{
|
||||||
|
gchar namebuf[1024];
|
||||||
|
guint16 priority, weight, port;
|
||||||
|
|
||||||
|
GETSHORT (priority, *p);
|
||||||
|
GETSHORT (weight, *p);
|
||||||
|
GETSHORT (port, *p);
|
||||||
|
*p += dn_expand (answer, end, *p, namebuf, sizeof (namebuf));
|
||||||
|
|
||||||
|
return g_variant_new ("(qqqs)",
|
||||||
|
priority,
|
||||||
|
weight,
|
||||||
|
port,
|
||||||
|
namebuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GVariant *
|
||||||
|
parse_res_soa (guchar *answer,
|
||||||
|
guchar *end,
|
||||||
|
guchar **p)
|
||||||
|
{
|
||||||
|
gchar mnamebuf[1024];
|
||||||
|
gchar rnamebuf[1024];
|
||||||
|
guint32 serial, refresh, retry, expire, ttl;
|
||||||
|
|
||||||
|
*p += dn_expand (answer, end, *p, mnamebuf, sizeof (mnamebuf));
|
||||||
|
*p += dn_expand (answer, end, *p, rnamebuf, sizeof (rnamebuf));
|
||||||
|
|
||||||
|
GETLONG (serial, *p);
|
||||||
|
GETLONG (refresh, *p);
|
||||||
|
GETLONG (retry, *p);
|
||||||
|
GETLONG (expire, *p);
|
||||||
|
GETLONG (ttl, *p);
|
||||||
|
|
||||||
|
return g_variant_new ("(ssuuuuu)",
|
||||||
|
mnamebuf,
|
||||||
|
rnamebuf,
|
||||||
|
serial,
|
||||||
|
refresh,
|
||||||
|
retry,
|
||||||
|
expire,
|
||||||
|
ttl);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GVariant *
|
||||||
|
parse_res_ns (guchar *answer,
|
||||||
|
guchar *end,
|
||||||
|
guchar **p)
|
||||||
|
{
|
||||||
|
gchar namebuf[1024];
|
||||||
|
|
||||||
|
*p += dn_expand (answer, end, *p, namebuf, sizeof (namebuf));
|
||||||
|
|
||||||
|
return g_variant_new ("(s)", namebuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GVariant *
|
||||||
|
parse_res_mx (guchar *answer,
|
||||||
|
guchar *end,
|
||||||
|
guchar **p)
|
||||||
|
{
|
||||||
|
gchar namebuf[1024];
|
||||||
|
guint16 preference;
|
||||||
|
|
||||||
|
GETSHORT (preference, *p);
|
||||||
|
|
||||||
|
*p += dn_expand (answer, end, *p, namebuf, sizeof (namebuf));
|
||||||
|
|
||||||
|
return g_variant_new ("(qs)",
|
||||||
|
preference,
|
||||||
|
namebuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GVariant *
|
||||||
|
parse_res_txt (guchar *answer,
|
||||||
|
guchar *end,
|
||||||
|
guchar **p)
|
||||||
|
{
|
||||||
|
GVariant *record;
|
||||||
|
GPtrArray *array;
|
||||||
|
guchar *at = *p;
|
||||||
|
gsize len;
|
||||||
|
|
||||||
|
array = g_ptr_array_new_with_free_func (g_free);
|
||||||
|
while (at < end)
|
||||||
|
{
|
||||||
|
len = *(at++);
|
||||||
|
if (len > at - end)
|
||||||
|
break;
|
||||||
|
g_ptr_array_add (array, g_strndup ((gchar *)at, len));
|
||||||
|
at += len;
|
||||||
|
}
|
||||||
|
|
||||||
|
*p = at;
|
||||||
|
record = g_variant_new ("(@as)",
|
||||||
|
g_variant_new_strv ((const gchar **)array->pdata, array->len));
|
||||||
|
g_ptr_array_free (array, TRUE);
|
||||||
|
return record;
|
||||||
|
}
|
||||||
|
|
||||||
|
gint
|
||||||
|
_g_resolver_record_type_to_rrtype (GResolverRecordType type)
|
||||||
|
{
|
||||||
|
switch (type)
|
||||||
|
{
|
||||||
|
case G_RESOLVER_RECORD_SRV:
|
||||||
|
return T_SRV;
|
||||||
|
case G_RESOLVER_RECORD_TXT:
|
||||||
|
return T_TXT;
|
||||||
|
case G_RESOLVER_RECORD_SOA:
|
||||||
|
return T_SOA;
|
||||||
|
case G_RESOLVER_RECORD_NS:
|
||||||
|
return T_NS;
|
||||||
|
case G_RESOLVER_RECORD_MX:
|
||||||
|
return T_MX;
|
||||||
|
}
|
||||||
|
g_return_val_if_reached (-1);
|
||||||
|
}
|
||||||
|
|
||||||
/* Private method to process a res_query response into GSrvTargets */
|
/* Private method to process a res_query response into GSrvTargets */
|
||||||
GList *
|
GList *
|
||||||
_g_resolver_targets_from_res_query (const gchar *rrname,
|
_g_resolver_records_from_res_query (const gchar *rrname,
|
||||||
|
gint rrtype,
|
||||||
guchar *answer,
|
guchar *answer,
|
||||||
gint len,
|
gint len,
|
||||||
gint herr,
|
gint herr,
|
||||||
@ -832,11 +1132,11 @@ _g_resolver_targets_from_res_query (const gchar *rrname,
|
|||||||
gint count;
|
gint count;
|
||||||
gchar namebuf[1024];
|
gchar namebuf[1024];
|
||||||
guchar *end, *p;
|
guchar *end, *p;
|
||||||
guint16 type, qclass, rdlength, priority, weight, port;
|
guint16 type, qclass, rdlength;
|
||||||
guint32 ttl;
|
guint32 ttl;
|
||||||
HEADER *header;
|
HEADER *header;
|
||||||
GSrvTarget *target;
|
GList *records;
|
||||||
GList *targets;
|
GVariant *record;
|
||||||
|
|
||||||
if (len <= 0)
|
if (len <= 0)
|
||||||
{
|
{
|
||||||
@ -846,7 +1146,7 @@ _g_resolver_targets_from_res_query (const gchar *rrname,
|
|||||||
if (len == 0 || herr == HOST_NOT_FOUND || herr == NO_DATA)
|
if (len == 0 || herr == HOST_NOT_FOUND || herr == NO_DATA)
|
||||||
{
|
{
|
||||||
errnum = G_RESOLVER_ERROR_NOT_FOUND;
|
errnum = G_RESOLVER_ERROR_NOT_FOUND;
|
||||||
format = _("No service record for '%s'");
|
format = _("No DNS record of the requested type for '%s'");
|
||||||
}
|
}
|
||||||
else if (herr == TRY_AGAIN)
|
else if (herr == TRY_AGAIN)
|
||||||
{
|
{
|
||||||
@ -863,7 +1163,7 @@ _g_resolver_targets_from_res_query (const gchar *rrname,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
targets = NULL;
|
records = NULL;
|
||||||
|
|
||||||
header = (HEADER *)answer;
|
header = (HEADER *)answer;
|
||||||
p = answer + sizeof (HEADER);
|
p = answer + sizeof (HEADER);
|
||||||
@ -875,6 +1175,9 @@ _g_resolver_targets_from_res_query (const gchar *rrname,
|
|||||||
{
|
{
|
||||||
p += dn_expand (answer, end, p, namebuf, sizeof (namebuf));
|
p += dn_expand (answer, end, p, namebuf, sizeof (namebuf));
|
||||||
p += 4;
|
p += 4;
|
||||||
|
|
||||||
|
/* To silence gcc warnings */
|
||||||
|
namebuf[0] = namebuf[1];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Read answers */
|
/* Read answers */
|
||||||
@ -888,34 +1191,126 @@ _g_resolver_targets_from_res_query (const gchar *rrname,
|
|||||||
ttl = ttl; /* To avoid -Wunused-but-set-variable */
|
ttl = ttl; /* To avoid -Wunused-but-set-variable */
|
||||||
GETSHORT (rdlength, p);
|
GETSHORT (rdlength, p);
|
||||||
|
|
||||||
if (type != T_SRV || qclass != C_IN)
|
if (type != rrtype || qclass != C_IN)
|
||||||
{
|
{
|
||||||
p += rdlength;
|
p += rdlength;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
GETSHORT (priority, p);
|
switch (rrtype)
|
||||||
GETSHORT (weight, p);
|
{
|
||||||
GETSHORT (port, p);
|
case T_SRV:
|
||||||
p += dn_expand (answer, end, p, namebuf, sizeof (namebuf));
|
record = parse_res_srv (answer, end, &p);
|
||||||
|
break;
|
||||||
target = g_srv_target_new (namebuf, port, priority, weight);
|
case T_MX:
|
||||||
targets = g_list_prepend (targets, target);
|
record = parse_res_mx (answer, end, &p);
|
||||||
|
break;
|
||||||
|
case T_SOA:
|
||||||
|
record = parse_res_soa (answer, end, &p);
|
||||||
|
break;
|
||||||
|
case T_NS:
|
||||||
|
record = parse_res_ns (answer, end, &p);
|
||||||
|
break;
|
||||||
|
case T_TXT:
|
||||||
|
record = parse_res_txt (answer, p + rdlength, &p);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
g_warn_if_reached ();
|
||||||
|
record = NULL;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return g_srv_target_list_sort (targets);
|
if (record != NULL)
|
||||||
|
records = g_list_prepend (records, record);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return records;
|
||||||
|
}
|
||||||
|
|
||||||
#elif defined(G_OS_WIN32)
|
#elif defined(G_OS_WIN32)
|
||||||
/* Private method to process a DnsQuery response into GSrvTargets */
|
static GVariant *
|
||||||
|
parse_dns_srv (DNS_RECORD *rec)
|
||||||
|
{
|
||||||
|
return g_variant_new ("(qqqs)",
|
||||||
|
(guint16)rec->Data.SRV.wPriority,
|
||||||
|
(guint16)rec->Data.SRV.wWeight,
|
||||||
|
(guint16)rec->Data.SRV.wPort,
|
||||||
|
rec->Data.SRV.pNameTarget);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GVariant *
|
||||||
|
parse_dns_soa (DNS_RECORD *rec)
|
||||||
|
{
|
||||||
|
return g_variant_new ("(ssuuuuu)",
|
||||||
|
rec->Data.SOA.pNamePrimaryServer,
|
||||||
|
rec->Data.SOA.pNameAdministrator,
|
||||||
|
(guint32)rec->Data.SOA.dwSerialNo,
|
||||||
|
(guint32)rec->Data.SOA.dwRefresh,
|
||||||
|
(guint32)rec->Data.SOA.dwRetry,
|
||||||
|
(guint32)rec->Data.SOA.dwExpire,
|
||||||
|
(guint32)rec->Data.SOA.dwDefaultTtl);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GVariant *
|
||||||
|
parse_dns_ns (DNS_RECORD *rec)
|
||||||
|
{
|
||||||
|
return g_variant_new ("(s)", rec->Data.NS.pNameHost);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GVariant *
|
||||||
|
parse_dns_mx (DNS_RECORD *rec)
|
||||||
|
{
|
||||||
|
return g_variant_new ("(qs)",
|
||||||
|
(guint16)rec->Data.MX.wPreference,
|
||||||
|
rec->Data.MX.pNameExchange);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GVariant *
|
||||||
|
parse_dns_txt (DNS_RECORD *rec)
|
||||||
|
{
|
||||||
|
GVariant *record;
|
||||||
|
GPtrArray *array;
|
||||||
|
DWORD i;
|
||||||
|
|
||||||
|
array = g_ptr_array_new ();
|
||||||
|
for (i = 0; i < rec->Data.TXT.dwStringCount; i++)
|
||||||
|
g_ptr_array_add (array, rec->Data.TXT.pStringArray[i]);
|
||||||
|
record = g_variant_new ("(@as)",
|
||||||
|
g_variant_new_strv ((const gchar **)array->pdata, array->len));
|
||||||
|
g_ptr_array_free (array, TRUE);
|
||||||
|
return record;
|
||||||
|
}
|
||||||
|
|
||||||
|
WORD
|
||||||
|
_g_resolver_record_type_to_dnstype (GResolverRecordType type)
|
||||||
|
{
|
||||||
|
switch (type)
|
||||||
|
{
|
||||||
|
case G_RESOLVER_RECORD_SRV:
|
||||||
|
return DNS_TYPE_SRV;
|
||||||
|
case G_RESOLVER_RECORD_TXT:
|
||||||
|
return DNS_TYPE_TEXT;
|
||||||
|
case G_RESOLVER_RECORD_SOA:
|
||||||
|
return DNS_TYPE_SOA;
|
||||||
|
case G_RESOLVER_RECORD_NS:
|
||||||
|
return DNS_TYPE_NS;
|
||||||
|
case G_RESOLVER_RECORD_MX:
|
||||||
|
return DNS_TYPE_MX;
|
||||||
|
}
|
||||||
|
g_return_val_if_reached (-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Private method to process a DnsQuery response into GVariants */
|
||||||
GList *
|
GList *
|
||||||
_g_resolver_targets_from_DnsQuery (const gchar *rrname,
|
_g_resolver_records_from_DnsQuery (const gchar *rrname,
|
||||||
|
WORD dnstype,
|
||||||
DNS_STATUS status,
|
DNS_STATUS status,
|
||||||
DNS_RECORD *results,
|
DNS_RECORD *results,
|
||||||
GError **error)
|
GError **error)
|
||||||
{
|
{
|
||||||
DNS_RECORD *rec;
|
DNS_RECORD *rec;
|
||||||
GSrvTarget *target;
|
gpointer record;
|
||||||
GList *targets;
|
GList *records;
|
||||||
|
|
||||||
if (status != ERROR_SUCCESS)
|
if (status != ERROR_SUCCESS)
|
||||||
{
|
{
|
||||||
@ -925,7 +1320,7 @@ _g_resolver_targets_from_DnsQuery (const gchar *rrname,
|
|||||||
if (status == DNS_ERROR_RCODE_NAME_ERROR)
|
if (status == DNS_ERROR_RCODE_NAME_ERROR)
|
||||||
{
|
{
|
||||||
errnum = G_RESOLVER_ERROR_NOT_FOUND;
|
errnum = G_RESOLVER_ERROR_NOT_FOUND;
|
||||||
format = _("No service record for '%s'");
|
format = _("No DNS record of the requested type for '%s'");
|
||||||
}
|
}
|
||||||
else if (status == DNS_ERROR_RCODE_SERVER_FAILURE)
|
else if (status == DNS_ERROR_RCODE_SERVER_FAILURE)
|
||||||
{
|
{
|
||||||
@ -942,20 +1337,38 @@ _g_resolver_targets_from_DnsQuery (const gchar *rrname,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
targets = NULL;
|
records = NULL;
|
||||||
for (rec = results; rec; rec = rec->pNext)
|
for (rec = results; rec; rec = rec->pNext)
|
||||||
{
|
{
|
||||||
if (rec->wType != DNS_TYPE_SRV)
|
if (rec->wType != dnstype)
|
||||||
continue;
|
continue;
|
||||||
|
switch (dnstype)
|
||||||
target = g_srv_target_new (rec->Data.SRV.pNameTarget,
|
{
|
||||||
rec->Data.SRV.wPort,
|
case DNS_TYPE_SRV:
|
||||||
rec->Data.SRV.wPriority,
|
record = parse_dns_srv (rec);
|
||||||
rec->Data.SRV.wWeight);
|
break;
|
||||||
targets = g_list_prepend (targets, target);
|
case DNS_TYPE_SOA:
|
||||||
|
record = parse_dns_soa (rec);
|
||||||
|
break;
|
||||||
|
case DNS_TYPE_NS:
|
||||||
|
record = parse_dns_ns (rec);
|
||||||
|
break;
|
||||||
|
case DNS_TYPE_MX:
|
||||||
|
record = parse_dns_mx (rec);
|
||||||
|
break;
|
||||||
|
case DNS_TYPE_TEXT:
|
||||||
|
record = parse_dns_txt (rec);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
g_warn_if_reached ();
|
||||||
|
record = NULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (record != NULL)
|
||||||
|
records = g_list_prepend (records, g_variant_ref_sink (record));
|
||||||
}
|
}
|
||||||
|
|
||||||
return g_srv_target_list_sort (targets);
|
return records;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -91,10 +91,24 @@ struct _GResolverClass {
|
|||||||
GAsyncResult *result,
|
GAsyncResult *result,
|
||||||
GError **error);
|
GError **error);
|
||||||
|
|
||||||
|
GList * ( *lookup_records) (GResolver *resolver,
|
||||||
|
const gchar *rrname,
|
||||||
|
GResolverRecordType record_type,
|
||||||
|
GCancellable *cancellable,
|
||||||
|
GError **error);
|
||||||
|
|
||||||
|
void ( *lookup_records_async) (GResolver *resolver,
|
||||||
|
const gchar *rrname,
|
||||||
|
GResolverRecordType record_type,
|
||||||
|
GCancellable *cancellable,
|
||||||
|
GAsyncReadyCallback callback,
|
||||||
|
gpointer user_data);
|
||||||
|
|
||||||
|
GList * ( *lookup_records_finish) (GResolver *resolver,
|
||||||
|
GAsyncResult *result,
|
||||||
|
GError **error);
|
||||||
|
|
||||||
/* Padding for future expansion */
|
/* Padding for future expansion */
|
||||||
void (*_g_reserved1) (void);
|
|
||||||
void (*_g_reserved2) (void);
|
|
||||||
void (*_g_reserved3) (void);
|
|
||||||
void (*_g_reserved4) (void);
|
void (*_g_reserved4) (void);
|
||||||
void (*_g_reserved5) (void);
|
void (*_g_reserved5) (void);
|
||||||
void (*_g_reserved6) (void);
|
void (*_g_reserved6) (void);
|
||||||
@ -150,6 +164,21 @@ GList *g_resolver_lookup_service_finish (GResolver *resolver,
|
|||||||
GAsyncResult *result,
|
GAsyncResult *result,
|
||||||
GError **error);
|
GError **error);
|
||||||
|
|
||||||
|
GList *g_resolver_lookup_records (GResolver *resolver,
|
||||||
|
const gchar *rrname,
|
||||||
|
GResolverRecordType record_type,
|
||||||
|
GCancellable *cancellable,
|
||||||
|
GError **error);
|
||||||
|
void g_resolver_lookup_records_async (GResolver *resolver,
|
||||||
|
const gchar *rrname,
|
||||||
|
GResolverRecordType record_type,
|
||||||
|
GCancellable *cancellable,
|
||||||
|
GAsyncReadyCallback callback,
|
||||||
|
gpointer user_data);
|
||||||
|
GList *g_resolver_lookup_records_finish (GResolver *resolver,
|
||||||
|
GAsyncResult *result,
|
||||||
|
GError **error);
|
||||||
|
|
||||||
void g_resolver_free_targets (GList *targets);
|
void g_resolver_free_targets (GList *targets);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -151,8 +151,9 @@ struct _GThreadedResolverRequest {
|
|||||||
} address;
|
} address;
|
||||||
struct {
|
struct {
|
||||||
gchar *rrname;
|
gchar *rrname;
|
||||||
GList *targets;
|
GResolverRecordType record_type;
|
||||||
} service;
|
GList *results;
|
||||||
|
} records;
|
||||||
} u;
|
} u;
|
||||||
|
|
||||||
GCancellable *cancellable;
|
GCancellable *cancellable;
|
||||||
@ -515,59 +516,81 @@ lookup_by_address_finish (GResolver *resolver,
|
|||||||
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
do_lookup_service (GThreadedResolverRequest *req,
|
do_lookup_records (GThreadedResolverRequest *req,
|
||||||
GError **error)
|
GError **error)
|
||||||
{
|
{
|
||||||
#if defined(G_OS_UNIX)
|
#if defined(G_OS_UNIX)
|
||||||
gint len, herr;
|
gint len = 512;
|
||||||
guchar answer[1024];
|
gint herr;
|
||||||
|
GByteArray *answer;
|
||||||
|
gint rrtype;
|
||||||
|
|
||||||
|
rrtype = _g_resolver_record_type_to_rrtype (req->u.records.record_type);
|
||||||
|
answer = g_byte_array_new ();
|
||||||
|
for (;;)
|
||||||
|
{
|
||||||
|
g_byte_array_set_size (answer, len * 2);
|
||||||
|
len = res_query (req->u.records.rrname, C_IN, rrtype, answer->data, answer->len);
|
||||||
|
|
||||||
|
/* If answer fit in the buffer then we're done */
|
||||||
|
if (len < 0 || len < (gint)answer->len)
|
||||||
|
break;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* On overflow some res_query's return the length needed, others
|
||||||
|
* return the full length entered. This code works in either case.
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
herr = h_errno;
|
||||||
|
req->u.records.results = _g_resolver_records_from_res_query (req->u.records.rrname, rrtype, answer->data, len, herr, error);
|
||||||
|
g_byte_array_free (answer, TRUE);
|
||||||
|
|
||||||
#elif defined(G_OS_WIN32)
|
#elif defined(G_OS_WIN32)
|
||||||
DNS_STATUS status;
|
DNS_STATUS status;
|
||||||
DNS_RECORD *results;
|
DNS_RECORD *results = NULL;
|
||||||
#endif
|
WORD dnstype;
|
||||||
|
|
||||||
#if defined(G_OS_UNIX)
|
dnstype = _g_resolver_record_type_to_dnstype (req->u.records.record_type);
|
||||||
len = res_query (req->u.service.rrname, C_IN, T_SRV, answer, sizeof (answer));
|
status = DnsQuery_A (req->u.records.rrname, dnstype, DNS_QUERY_STANDARD, NULL, &results, NULL);
|
||||||
herr = h_errno;
|
req->u.records.results = _g_resolver_records_from_DnsQuery (req->u.records.rrname, dnstype, status, results, error);
|
||||||
req->u.service.targets = _g_resolver_targets_from_res_query (req->u.service.rrname, answer, len, herr, error);
|
if (results != NULL)
|
||||||
#elif defined(G_OS_WIN32)
|
|
||||||
status = DnsQuery_A (req->u.service.rrname, DNS_TYPE_SRV,
|
|
||||||
DNS_QUERY_STANDARD, NULL, &results, NULL);
|
|
||||||
req->u.service.targets = _g_resolver_targets_from_DnsQuery (req->u.service.rrname, status, results, error);
|
|
||||||
DnsRecordListFree (results, DnsFreeRecordList);
|
DnsRecordListFree (results, DnsFreeRecordList);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
static GList *
|
static GList *
|
||||||
lookup_service (GResolver *resolver,
|
lookup_records (GResolver *resolver,
|
||||||
const gchar *rrname,
|
const gchar *rrname,
|
||||||
|
GResolverRecordType record_type,
|
||||||
GCancellable *cancellable,
|
GCancellable *cancellable,
|
||||||
GError **error)
|
GError **error)
|
||||||
{
|
{
|
||||||
GThreadedResolver *gtr = G_THREADED_RESOLVER (resolver);
|
GThreadedResolver *gtr = G_THREADED_RESOLVER (resolver);
|
||||||
GThreadedResolverRequest *req;
|
GThreadedResolverRequest *req;
|
||||||
GList *targets;
|
GList *results;
|
||||||
|
|
||||||
req = g_threaded_resolver_request_new (do_lookup_service, NULL, cancellable);
|
req = g_threaded_resolver_request_new (do_lookup_records, NULL, cancellable);
|
||||||
req->u.service.rrname = (char *)rrname;
|
req->u.records.rrname = (char *)rrname;
|
||||||
|
req->u.records.record_type = record_type;
|
||||||
resolve_sync (gtr, req, error);
|
resolve_sync (gtr, req, error);
|
||||||
|
|
||||||
targets = req->u.service.targets;
|
results = req->u.records.results;
|
||||||
g_threaded_resolver_request_unref (req);
|
g_threaded_resolver_request_unref (req);
|
||||||
return targets;
|
return results;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
free_lookup_service (GThreadedResolverRequest *req)
|
free_lookup_records (GThreadedResolverRequest *req)
|
||||||
{
|
{
|
||||||
g_free (req->u.service.rrname);
|
g_free (req->u.records.rrname);
|
||||||
if (req->u.service.targets)
|
g_list_free_full (req->u.records.results, (GDestroyNotify)g_variant_unref);
|
||||||
g_resolver_free_targets (req->u.service.targets);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
lookup_service_async (GResolver *resolver,
|
lookup_records_async (GResolver *resolver,
|
||||||
const char *rrname,
|
const char *rrname,
|
||||||
|
GResolverRecordType record_type,
|
||||||
GCancellable *cancellable,
|
GCancellable *cancellable,
|
||||||
GAsyncReadyCallback callback,
|
GAsyncReadyCallback callback,
|
||||||
gpointer user_data)
|
gpointer user_data)
|
||||||
@ -575,25 +598,26 @@ lookup_service_async (GResolver *resolver,
|
|||||||
GThreadedResolver *gtr = G_THREADED_RESOLVER (resolver);
|
GThreadedResolver *gtr = G_THREADED_RESOLVER (resolver);
|
||||||
GThreadedResolverRequest *req;
|
GThreadedResolverRequest *req;
|
||||||
|
|
||||||
req = g_threaded_resolver_request_new (do_lookup_service,
|
req = g_threaded_resolver_request_new (do_lookup_records,
|
||||||
free_lookup_service,
|
free_lookup_records,
|
||||||
cancellable);
|
cancellable);
|
||||||
req->u.service.rrname = g_strdup (rrname);
|
req->u.records.rrname = g_strdup (rrname);
|
||||||
resolve_async (gtr, req, callback, user_data, lookup_service_async);
|
req->u.records.record_type = record_type;
|
||||||
|
resolve_async (gtr, req, callback, user_data, lookup_records_async);
|
||||||
}
|
}
|
||||||
|
|
||||||
static GList *
|
static GList *
|
||||||
lookup_service_finish (GResolver *resolver,
|
lookup_records_finish (GResolver *resolver,
|
||||||
GAsyncResult *result,
|
GAsyncResult *result,
|
||||||
GError **error)
|
GError **error)
|
||||||
{
|
{
|
||||||
GThreadedResolverRequest *req;
|
GThreadedResolverRequest *req;
|
||||||
GList *targets;
|
GList *records;
|
||||||
|
|
||||||
req = resolve_finish (resolver, result, lookup_service_async, error);
|
req = resolve_finish (resolver, result, lookup_records_async, error);
|
||||||
targets = req->u.service.targets;
|
records = req->u.records.results;
|
||||||
req->u.service.targets = NULL;
|
req->u.records.results = NULL;
|
||||||
return targets;
|
return records;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -609,9 +633,9 @@ g_threaded_resolver_class_init (GThreadedResolverClass *threaded_class)
|
|||||||
resolver_class->lookup_by_address = lookup_by_address;
|
resolver_class->lookup_by_address = lookup_by_address;
|
||||||
resolver_class->lookup_by_address_async = lookup_by_address_async;
|
resolver_class->lookup_by_address_async = lookup_by_address_async;
|
||||||
resolver_class->lookup_by_address_finish = lookup_by_address_finish;
|
resolver_class->lookup_by_address_finish = lookup_by_address_finish;
|
||||||
resolver_class->lookup_service = lookup_service;
|
resolver_class->lookup_records = lookup_records;
|
||||||
resolver_class->lookup_service_async = lookup_service_async;
|
resolver_class->lookup_records_async = lookup_records_async;
|
||||||
resolver_class->lookup_service_finish = lookup_service_finish;
|
resolver_class->lookup_records_finish = lookup_records_finish;
|
||||||
|
|
||||||
object_class->finalize = finalize;
|
object_class->finalize = finalize;
|
||||||
}
|
}
|
||||||
|
@ -36,15 +36,20 @@ static GResolver *resolver;
|
|||||||
static GCancellable *cancellable;
|
static GCancellable *cancellable;
|
||||||
static GMainLoop *loop;
|
static GMainLoop *loop;
|
||||||
static int nlookups = 0;
|
static int nlookups = 0;
|
||||||
|
static gboolean synchronous = FALSE;
|
||||||
|
static guint connectable_count = 0;
|
||||||
|
static GResolverRecordType record_type = 0;
|
||||||
|
|
||||||
static void G_GNUC_NORETURN
|
static void G_GNUC_NORETURN
|
||||||
usage (void)
|
usage (void)
|
||||||
{
|
{
|
||||||
fprintf (stderr, "Usage: resolver [-s] [hostname | IP | service/protocol/domain ] ...\n");
|
fprintf (stderr, "Usage: resolver [-s] [hostname | IP | service/protocol/domain ] ...\n");
|
||||||
|
fprintf (stderr, "Usage: resolver [-s] [-t MX|TXT|NS|SOA] rrname ...\n");
|
||||||
fprintf (stderr, " resolver [-s] -c NUMBER [hostname | IP | service/protocol/domain ]\n");
|
fprintf (stderr, " resolver [-s] -c NUMBER [hostname | IP | service/protocol/domain ]\n");
|
||||||
fprintf (stderr, " Use -s to do synchronous lookups.\n");
|
fprintf (stderr, " Use -s to do synchronous lookups.\n");
|
||||||
fprintf (stderr, " Use -c NUMBER (and only a single resolvable argument) to test GSocketConnectable.\n");
|
fprintf (stderr, " Use -c NUMBER (and only a single resolvable argument) to test GSocketConnectable.\n");
|
||||||
fprintf (stderr, " The given NUMBER determines how many times the connectable will be enumerated.\n");
|
fprintf (stderr, " The given NUMBER determines how many times the connectable will be enumerated.\n");
|
||||||
|
fprintf (stderr, " Use -t with MX, TXT, NS or SOA to lookup DNS records of those types.\n");
|
||||||
exit (1);
|
exit (1);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -138,9 +143,9 @@ print_resolved_service (const char *service,
|
|||||||
{
|
{
|
||||||
printf ("%s:%u (pri %u, weight %u)\n",
|
printf ("%s:%u (pri %u, weight %u)\n",
|
||||||
g_srv_target_get_hostname (t->data),
|
g_srv_target_get_hostname (t->data),
|
||||||
g_srv_target_get_port (t->data),
|
(guint)g_srv_target_get_port (t->data),
|
||||||
g_srv_target_get_priority (t->data),
|
(guint)g_srv_target_get_priority (t->data),
|
||||||
g_srv_target_get_weight (t->data));
|
(guint)g_srv_target_get_weight (t->data));
|
||||||
g_srv_target_free (t->data);
|
g_srv_target_free (t->data);
|
||||||
}
|
}
|
||||||
g_list_free (targets);
|
g_list_free (targets);
|
||||||
@ -151,12 +156,186 @@ print_resolved_service (const char *service,
|
|||||||
G_UNLOCK (response);
|
G_UNLOCK (response);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
print_resolved_mx (const char *rrname,
|
||||||
|
GList *records,
|
||||||
|
GError *error)
|
||||||
|
{
|
||||||
|
const gchar *hostname;
|
||||||
|
guint16 priority;
|
||||||
|
GList *t;
|
||||||
|
|
||||||
|
G_LOCK (response);
|
||||||
|
printf ("Domain: %s\n", rrname);
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
printf ("Error: %s\n", error->message);
|
||||||
|
g_error_free (error);
|
||||||
|
}
|
||||||
|
else if (!records)
|
||||||
|
{
|
||||||
|
printf ("no MX records\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (t = records; t; t = t->next)
|
||||||
|
{
|
||||||
|
g_variant_get (t->data, "(q&s)", &priority, &hostname);
|
||||||
|
printf ("%s (pri %u)\n", hostname, (guint)priority);
|
||||||
|
g_variant_unref (t->data);
|
||||||
|
}
|
||||||
|
g_list_free (records);
|
||||||
|
}
|
||||||
|
printf ("\n");
|
||||||
|
|
||||||
|
done_lookup ();
|
||||||
|
G_UNLOCK (response);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
print_resolved_txt (const char *rrname,
|
||||||
|
GList *records,
|
||||||
|
GError *error)
|
||||||
|
{
|
||||||
|
const gchar **contents;
|
||||||
|
GList *t;
|
||||||
|
gint i;
|
||||||
|
|
||||||
|
G_LOCK (response);
|
||||||
|
printf ("Domain: %s\n", rrname);
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
printf ("Error: %s\n", error->message);
|
||||||
|
g_error_free (error);
|
||||||
|
}
|
||||||
|
else if (!records)
|
||||||
|
{
|
||||||
|
printf ("no TXT records\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (t = records; t; t = t->next)
|
||||||
|
{
|
||||||
|
if (t != records)
|
||||||
|
printf ("\n");
|
||||||
|
g_variant_get (t->data, "(^a&s)", &contents);
|
||||||
|
for (i = 0; contents[i] != NULL; i++)
|
||||||
|
printf ("%s\n", contents[i]);
|
||||||
|
g_variant_unref (t->data);
|
||||||
|
}
|
||||||
|
g_list_free (records);
|
||||||
|
}
|
||||||
|
printf ("\n");
|
||||||
|
|
||||||
|
done_lookup ();
|
||||||
|
G_UNLOCK (response);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
print_resolved_soa (const char *rrname,
|
||||||
|
GList *records,
|
||||||
|
GError *error)
|
||||||
|
{
|
||||||
|
GList *t;
|
||||||
|
const gchar *primary_ns;
|
||||||
|
const gchar *administrator;
|
||||||
|
guint32 serial, refresh, retry, expire, ttl;
|
||||||
|
|
||||||
|
G_LOCK (response);
|
||||||
|
printf ("Zone: %s\n", rrname);
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
printf ("Error: %s\n", error->message);
|
||||||
|
g_error_free (error);
|
||||||
|
}
|
||||||
|
else if (!records)
|
||||||
|
{
|
||||||
|
printf ("no SOA records\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (t = records; t; t = t->next)
|
||||||
|
{
|
||||||
|
g_variant_get (t->data, "(&s&suuuuu)", &primary_ns, &administrator,
|
||||||
|
&serial, &refresh, &retry, &expire, &ttl);
|
||||||
|
printf ("%s %s (serial %u, refresh %u, retry %u, expire %u, ttl %u)\n",
|
||||||
|
primary_ns, administrator, (guint)serial, (guint)refresh,
|
||||||
|
(guint)retry, (guint)expire, (guint)ttl);
|
||||||
|
g_variant_unref (t->data);
|
||||||
|
}
|
||||||
|
g_list_free (records);
|
||||||
|
}
|
||||||
|
printf ("\n");
|
||||||
|
|
||||||
|
done_lookup ();
|
||||||
|
G_UNLOCK (response);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
print_resolved_ns (const char *rrname,
|
||||||
|
GList *records,
|
||||||
|
GError *error)
|
||||||
|
{
|
||||||
|
GList *t;
|
||||||
|
const gchar *hostname;
|
||||||
|
|
||||||
|
G_LOCK (response);
|
||||||
|
printf ("Zone: %s\n", rrname);
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
printf ("Error: %s\n", error->message);
|
||||||
|
g_error_free (error);
|
||||||
|
}
|
||||||
|
else if (!records)
|
||||||
|
{
|
||||||
|
printf ("no NS records\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (t = records; t; t = t->next)
|
||||||
|
{
|
||||||
|
g_variant_get (t->data, "(&s)", &hostname);
|
||||||
|
printf ("%s\n", hostname);
|
||||||
|
g_variant_unref (t->data);
|
||||||
|
}
|
||||||
|
g_list_free (records);
|
||||||
|
}
|
||||||
|
printf ("\n");
|
||||||
|
|
||||||
|
done_lookup ();
|
||||||
|
G_UNLOCK (response);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
lookup_one_sync (const char *arg)
|
lookup_one_sync (const char *arg)
|
||||||
{
|
{
|
||||||
GError *error = NULL;
|
GError *error = NULL;
|
||||||
|
|
||||||
if (strchr (arg, '/'))
|
if (record_type != 0)
|
||||||
|
{
|
||||||
|
GList *records;
|
||||||
|
|
||||||
|
records = g_resolver_lookup_records (resolver, arg, record_type, cancellable, &error);
|
||||||
|
switch (record_type)
|
||||||
|
{
|
||||||
|
case G_RESOLVER_RECORD_MX:
|
||||||
|
print_resolved_mx (arg, records, error);
|
||||||
|
break;
|
||||||
|
case G_RESOLVER_RECORD_SOA:
|
||||||
|
print_resolved_soa (arg, records, error);
|
||||||
|
break;
|
||||||
|
case G_RESOLVER_RECORD_NS:
|
||||||
|
print_resolved_ns (arg, records, error);
|
||||||
|
break;
|
||||||
|
case G_RESOLVER_RECORD_TXT:
|
||||||
|
print_resolved_txt (arg, records, error);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
g_warn_if_reached ();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (strchr (arg, '/'))
|
||||||
{
|
{
|
||||||
GList *targets;
|
GList *targets;
|
||||||
/* service/protocol/domain */
|
/* service/protocol/domain */
|
||||||
@ -244,6 +423,37 @@ lookup_service_callback (GObject *source, GAsyncResult *result,
|
|||||||
print_resolved_service (service, targets, error);
|
print_resolved_service (service, targets, error);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
lookup_records_callback (GObject *source,
|
||||||
|
GAsyncResult *result,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
const char *arg = user_data;
|
||||||
|
GError *error = NULL;
|
||||||
|
GList *records;
|
||||||
|
|
||||||
|
records = g_resolver_lookup_records_finish (resolver, result, &error);
|
||||||
|
|
||||||
|
switch (record_type)
|
||||||
|
{
|
||||||
|
case G_RESOLVER_RECORD_MX:
|
||||||
|
print_resolved_mx (arg, records, error);
|
||||||
|
break;
|
||||||
|
case G_RESOLVER_RECORD_SOA:
|
||||||
|
print_resolved_soa (arg, records, error);
|
||||||
|
break;
|
||||||
|
case G_RESOLVER_RECORD_NS:
|
||||||
|
print_resolved_ns (arg, records, error);
|
||||||
|
break;
|
||||||
|
case G_RESOLVER_RECORD_TXT:
|
||||||
|
print_resolved_txt (arg, records, error);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
g_warn_if_reached ();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
start_async_lookups (char **argv, int argc)
|
start_async_lookups (char **argv, int argc)
|
||||||
{
|
{
|
||||||
@ -251,7 +461,12 @@ start_async_lookups (char **argv, int argc)
|
|||||||
|
|
||||||
for (i = 0; i < argc; i++)
|
for (i = 0; i < argc; i++)
|
||||||
{
|
{
|
||||||
if (strchr (argv[i], '/'))
|
if (record_type != 0)
|
||||||
|
{
|
||||||
|
g_resolver_lookup_records_async (resolver, argv[i], record_type,
|
||||||
|
cancellable, lookup_records_callback, argv[i]);
|
||||||
|
}
|
||||||
|
else if (strchr (argv[i], '/'))
|
||||||
{
|
{
|
||||||
/* service/protocol/domain */
|
/* service/protocol/domain */
|
||||||
char **parts = g_strsplit (argv[i], "/", 3);
|
char **parts = g_strsplit (argv[i], "/", 3);
|
||||||
@ -428,11 +643,42 @@ async_cancel (GIOChannel *source, GIOCondition cond, gpointer cancel)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
record_type_arg (const gchar *option_name,
|
||||||
|
const gchar *value,
|
||||||
|
gpointer data,
|
||||||
|
GError **error)
|
||||||
|
{
|
||||||
|
if (g_ascii_strcasecmp (value, "MX") == 0) {
|
||||||
|
record_type = G_RESOLVER_RECORD_MX;
|
||||||
|
} else if (g_ascii_strcasecmp (value, "TXT") == 0) {
|
||||||
|
record_type = G_RESOLVER_RECORD_TXT;
|
||||||
|
} else if (g_ascii_strcasecmp (value, "SOA") == 0) {
|
||||||
|
record_type = G_RESOLVER_RECORD_SOA;
|
||||||
|
} else if (g_ascii_strcasecmp (value, "NS") == 0) {
|
||||||
|
record_type = G_RESOLVER_RECORD_NS;
|
||||||
|
} else {
|
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE,
|
||||||
|
"Specify MX, TXT, NS or SOA for the special record lookup types");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const GOptionEntry option_entries[] = {
|
||||||
|
{ "synchronous", 's', 0, G_OPTION_ARG_NONE, &synchronous, "Synchronous connections", NULL },
|
||||||
|
{ "connectable", 'c', 0, G_OPTION_ARG_INT, &connectable_count, "Connectable count", "C" },
|
||||||
|
{ "special-type", 't', 0, G_OPTION_ARG_CALLBACK, record_type_arg, "Record type like MX, TXT, NS or SOA", "RR" },
|
||||||
|
{ NULL },
|
||||||
|
};
|
||||||
|
|
||||||
int
|
int
|
||||||
main (int argc, char **argv)
|
main (int argc, char **argv)
|
||||||
{
|
{
|
||||||
gboolean synchronous = FALSE;
|
GOptionContext *context;
|
||||||
guint connectable_count = 0;
|
GError *error = NULL;
|
||||||
#ifdef G_OS_UNIX
|
#ifdef G_OS_UNIX
|
||||||
GIOChannel *chan;
|
GIOChannel *chan;
|
||||||
guint watch;
|
guint watch;
|
||||||
@ -440,22 +686,13 @@ main (int argc, char **argv)
|
|||||||
|
|
||||||
g_type_init ();
|
g_type_init ();
|
||||||
|
|
||||||
/* FIXME: use GOptionContext */
|
context = g_option_context_new ("lookups ...");
|
||||||
while (argc >= 2 && argv[1][0] == '-')
|
g_option_context_add_main_entries (context, option_entries, NULL);
|
||||||
|
if (!g_option_context_parse (context, &argc, &argv, &error))
|
||||||
{
|
{
|
||||||
if (!strcmp (argv[1], "-s"))
|
g_printerr ("%s\n", error->message);
|
||||||
synchronous = TRUE;
|
g_error_free (error);
|
||||||
else if (!strcmp (argv[1], "-c"))
|
|
||||||
{
|
|
||||||
connectable_count = atoi (argv[2]);
|
|
||||||
argv++;
|
|
||||||
argc--;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
usage();
|
usage();
|
||||||
|
|
||||||
argv++;
|
|
||||||
argc--;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (argc < 2 || (argc > 2 && connectable_count))
|
if (argc < 2 || (argc > 2 && connectable_count))
|
||||||
|
Loading…
Reference in New Issue
Block a user