Pulling MX record from DNS server

15,471

Solution 1

The simplest method is to simply use commonly available tools.

The basic "dig" command will return the records to you via this query:

dig mx example.com

If you want just the lines with the mx records...

dig mx example.com | grep -v '^;' | grep example.com

dig is available on most linux / unix boxes.

If you're on windows you can use nslookup

nslookup -type=mx example.com

Then just parse the output of these common tools.

EDIT: Simple C example of sockets from the web

Since you put "C" as a tag, I guess you're looking for source code to do MX lookups using raw sockets. I copied this from http://www.developerweb.net/forum/showthread.php?t=3550. It may be more what you're looking for?

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <netinet/in.h>
#include <resolv.h>

int main (int argc, char *argv[])
{
    u_char nsbuf[4096];
    char dispbuf[4096];
    ns_msg msg;
    ns_rr rr;
    int i, j, l;

    if (argc < 2) {
        printf ("Usage: %s <domain>[...]\n", argv[0]);
        exit (1);
    }

    for (i = 1; i < argc; i++) {
        l = res_query (argv[i], ns_c_any, ns_t_mx, nsbuf, sizeof (nsbuf));
        if (l < 0) {
            perror (argv[i]);
        } else {
#ifdef USE_PQUERY
/* this will give lots of detailed info on the request and reply */
            res_pquery (&_res, nsbuf, l, stdout);
#else
/* just grab the MX answer info */
            ns_initparse (nsbuf, l, &msg);
            printf ("%s :\n", argv[i]);
            l = ns_msg_count (msg, ns_s_an);
            for (j = 0; j < l; j++) {
                ns_parserr (&msg, ns_s_an, j, &rr);
                ns_sprintrr (&msg, &rr, NULL, NULL, dispbuf, sizeof (dispbuf));
                printf ("%s\n", dispbuf);
            }
#endif
        }
    }

    exit (0);
}

Solution 2

on linux:

host -t mx google.com

Solution 3

I notice that you're writing for Linux. The idomatic way for a regular program to send mail on Unix-like systems is either:

  • Run /usr/bin/mail in a subprocess and send it the mail message on its standard input (see the mail manpage); or
  • Connect to 127.0.0.1:25 and give the local mail daemon the message to deliver.

Both ways presume that the local mailer is configured to pass mail on to where it has to go; on a well-configured Linux box this is a fair assumption.

If that doesn't appeal, the second-best way is for your program to accept the address of a local mail relay server to use, and just connect to that server on port 25.

In other words, wherever possible, use an existing mail relay to send your mail on. Those mail relays will have all the local knowledge that might be necessary to get mail out of the network that you're running on - just looking up the MX and trying to send directly to the destination is not always going to work.

If you've read all that and you still want to look up MX records, try the adns library, it takes care of all the tedious details involved in DNS resolution (and believe me, it is tedious, and easy to get wrong!).

Share:
15,471
Suroot
Author by

Suroot

I'm a big fan of C/C++/C#/VHDL and moreover DirectX, HLSL, OpenGL programming. Been doing Gtk, Gtk+2.0, Qt3.3, Qt4.0+.

Updated on June 06, 2022

Comments

  • Suroot
    Suroot almost 2 years

    I am writing an application that is requiring me to do a DNS lookup for an MX record. I'm not sure if anyone has had experience doing this kind of work but if you do, any help would be appreciated.

    EDIT: The thing that I'm going for is an application that will send an e-mail alert. The problem is I need to have the application be able to lookup the MX record for a domain.

  • Suroot
    Suroot almost 15 years
    Sorry for the confusion, I'm trying to find the code that would be used in dig to find the MX record. I know that dig and nslookup exist, my thing is that I want to write an application that will do these lookups without an external application.
  • Greg Rogers
    Greg Rogers almost 15 years
    You can download the source code for it here: isc.org/downloadables/11
  • Suroot
    Suroot almost 15 years
    Thanks, the I'm checking out the code you had posted a bit ago, seems relatively close and more importantly the resolver(3) man page seems pretty useful there.
  • andrew
    andrew almost 15 years
    I reverted to the answer that included the code since you thought it was useful.
  • Mike S
    Mike S over 3 years
    I know it's late :-), but I've had occasion to bump into this question in 2021. You may want to replace res_query with res_search. The latter will expand shortnames to the paths listed in the "search" entry in resolv.conf.