// Copyright (c) 2011 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "net/base/dns_query.h"

#include "base/bind.h"
#include "base/rand_util.h"
#include "net/base/dns_util.h"
#include "testing/gtest/include/gtest/gtest.h"

namespace net {

// DNS query consists of a header followed by a question.  Header format
// and question format are described below.  For the meaning of specific
// fields, please see RFC 1035.

// Header format.
//                                  1  1  1  1  1  1
//    0  1  2  3  4  5  6  7  8  9  0  1  2  3  4  5
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
//  |                      ID                       |
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
//  |QR|   Opcode  |AA|TC|RD|RA|   Z    |   RCODE   |
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
//  |                    QDCOUNT                    |
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
//  |                    ANCOUNT                    |
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
//  |                    NSCOUNT                    |
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
//  |                    ARCOUNT                    |
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+

// Question format.
//                                  1  1  1  1  1  1
//    0  1  2  3  4  5  6  7  8  9  0  1  2  3  4  5
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
//  |                                               |
//  /                     QNAME                     /
//  /                                               /
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
//  |                     QTYPE                     |
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
//  |                     QCLASS                    |
//  +--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+

TEST(DnsQueryTest, ConstructorTest) {
  std::string kQname("\003www\006google\003com", 16);
  DnsQuery q1(kQname, kDNS_A, base::Bind(&base::RandInt));
  EXPECT_EQ(kDNS_A, q1.qtype());

  uint8 id_hi = q1.id() >> 8, id_lo = q1.id() & 0xff;

  // See the top of the file for the description of a DNS query.
  const uint8 query_data[] = {
    // Header
    id_hi, id_lo,
    0x01, 0x00,               // Flags -- set RD (recursion desired) bit.
    0x00, 0x01,               // Set QDCOUNT (question count) to 1, all the
                              // rest are 0 for a query.
    0x00, 0x00,
    0x00, 0x00,
    0x00, 0x00,

    // Question
    0x03, 0x77, 0x77, 0x77,   // QNAME: www.google.com in DNS format.
    0x06, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65,
    0x03, 0x63, 0x6f, 0x6d, 0x00,

    0x00, 0x01,               // QTYPE: A query.
    0x00, 0x01,               // QCLASS: IN class.
  };

  int expected_size = arraysize(query_data);
  EXPECT_EQ(expected_size, q1.io_buffer()->size());
  EXPECT_EQ(0, memcmp(q1.io_buffer()->data(), query_data, expected_size));
}

TEST(DnsQueryTest, CloneTest) {
  std::string kQname("\003www\006google\003com", 16);
  DnsQuery q1(kQname, kDNS_A, base::Bind(&base::RandInt));

  scoped_ptr<DnsQuery> q2(q1.CloneWithNewId());
  EXPECT_EQ(q1.io_buffer()->size(), q2->io_buffer()->size());
  EXPECT_EQ(q1.qtype(), q2->qtype());
  EXPECT_EQ(q1.question_size(), q2->question_size());
  EXPECT_EQ(0, memcmp(q1.question_data(), q2->question_data(),
                      q1.question_size()));
}

TEST(DnsQueryTest, RandomIdTest) {
  std::string kQname("\003www\006google\003com", 16);

  // Since id fields are 16-bit values, we iterate to reduce the
  // probability of collision, to avoid a flaky test.
  bool ids_are_random = false;
  for (int i = 0; i < 1000; ++i) {
    DnsQuery q1(kQname, kDNS_A, base::Bind(&base::RandInt));
    DnsQuery q2(kQname, kDNS_A, base::Bind(&base::RandInt));
    scoped_ptr<DnsQuery> q3(q1.CloneWithNewId());
    ids_are_random = q1.id () != q2.id() && q1.id() != q3->id();
    if (ids_are_random)
      break;
  }
  EXPECT_TRUE(ids_are_random);
}

}  // namespace net