123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207 |
- // Copyright (C) 2014 Internet Systems Consortium, Inc. ("ISC")
- //
- // Permission to use, copy, modify, and/or distribute this software for any
- // purpose with or without fee is hereby granted, provided that the above
- // copyright notice and this permission notice appear in all copies.
- //
- // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
- // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
- // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
- // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
- // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
- // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
- // PERFORMANCE OF THIS SOFTWARE.
- #include <dhcp_ddns/watch_socket.h>
- #include <test_utils.h>
- #include <gtest/gtest.h>
- #include <sys/select.h>
- #include <sys/ioctl.h>
- using namespace std;
- using namespace isc;
- using namespace isc::dhcp_ddns;
- namespace {
- /// @brief Tests the basic functionality of WatchSocket.
- TEST(WatchSocketTest, basics) {
- WatchSocketPtr watch;
- /// Verify that we can construct a WatchSocket.
- ASSERT_NO_THROW(watch.reset(new WatchSocket()));
- ASSERT_TRUE(watch);
- /// Verify that post-construction the state the select-fd is valid.
- int select_fd = watch->getSelectFd();
- EXPECT_NE(select_fd, WatchSocket::INVALID_SOCKET);
- /// Verify that isReady() is false and that a call to select agrees.
- EXPECT_FALSE(watch->isReady());
- EXPECT_EQ(0, selectCheck(select_fd));
- /// Verify that the socket can be marked ready.
- ASSERT_NO_THROW(watch->markReady());
- /// Verify that we have exactly one marker waiting to be read.
- int count = 0;
- EXPECT_FALSE(ioctl(select_fd, FIONREAD, &count));
- EXPECT_EQ(sizeof(WatchSocket::MARKER), count);
- /// Verify that we can call markReady again without error.
- ASSERT_NO_THROW(watch->markReady());
- /// Verify that we STILL have exactly one marker waiting to be read.
- EXPECT_FALSE(ioctl(select_fd, FIONREAD, &count));
- EXPECT_EQ(sizeof(WatchSocket::MARKER), count);
- /// Verify that isReady() is true and that a call to select agrees.
- EXPECT_TRUE(watch->isReady());
- EXPECT_EQ(1, selectCheck(select_fd));
- /// Verify that the socket can be cleared.
- ASSERT_NO_THROW(watch->clearReady());
- /// Verify that isReady() is false and that a call to select agrees.
- EXPECT_FALSE(watch->isReady());
- EXPECT_EQ(0, selectCheck(select_fd));
- }
- /// @brief Checks behavior when select_fd is closed externally while in the
- /// "cleared" state.
- TEST(WatchSocketTest, closedWhileClear) {
- WatchSocketPtr watch;
- /// Verify that we can construct a WatchSocket.
- ASSERT_NO_THROW(watch.reset(new WatchSocket()));
- ASSERT_TRUE(watch);
- /// Verify that post-construction the state the select-fd is valid.
- int select_fd = watch->getSelectFd();
- ASSERT_NE(select_fd, WatchSocket::INVALID_SOCKET);
- // Verify that socket does not appear ready.
- ASSERT_EQ(0, watch->isReady());
- // Interfere by closing the fd.
- ASSERT_EQ(0, close(select_fd));
- // Verify that socket does not appear ready.
- ASSERT_EQ(0, watch->isReady());
- // Verify that clear does NOT throw.
- ASSERT_NO_THROW(watch->clearReady());
- // Verify that trying to mark it fails.
- ASSERT_THROW(watch->markReady(), WatchSocketError);
- // Verify that clear does NOT throw.
- ASSERT_NO_THROW(watch->clearReady());
- // Verify that getSelectFd() returns invalid socket.
- ASSERT_EQ(WatchSocket::INVALID_SOCKET, watch->getSelectFd());
- }
- /// @brief Checks behavior when select_fd has closed while in the "ready"
- /// state.
- TEST(WatchSocketTest, closedWhileReady) {
- WatchSocketPtr watch;
- /// Verify that we can construct a WatchSocket.
- ASSERT_NO_THROW(watch.reset(new WatchSocket()));
- ASSERT_TRUE(watch);
- /// Verify that post-construction the state the select-fd is valid.
- int select_fd = watch->getSelectFd();
- ASSERT_NE(select_fd, WatchSocket::INVALID_SOCKET);
- /// Verify that the socket can be marked ready.
- ASSERT_NO_THROW(watch->markReady());
- EXPECT_EQ(1, selectCheck(select_fd));
- // Interfere by closing the fd.
- ASSERT_EQ(0, close(select_fd));
- // Verify that trying to clear it does not throw.
- ASSERT_NO_THROW(watch->clearReady());
- // Verify the select_fd fails as socket is invalid/closed.
- EXPECT_EQ(-1, selectCheck(select_fd));
- // Verify that subsequent attempts to mark it will fail.
- ASSERT_THROW(watch->markReady(), WatchSocketError);
- }
- /// @brief Checks behavior when select_fd has been marked ready but then
- /// emptied by an external read.
- TEST(WatchSocketTest, emptyReadySelectFd) {
- WatchSocketPtr watch;
- /// Verify that we can construct a WatchSocket.
- ASSERT_NO_THROW(watch.reset(new WatchSocket()));
- ASSERT_TRUE(watch);
- /// Verify that post-construction the state the select-fd is valid.
- int select_fd = watch->getSelectFd();
- ASSERT_NE(select_fd, WatchSocket::INVALID_SOCKET);
- /// Verify that the socket can be marked ready.
- ASSERT_NO_THROW(watch->markReady());
- EXPECT_EQ(1, selectCheck(select_fd));
- // Interfere by reading the fd. This should empty the read pipe.
- uint32_t buf = 0;
- ASSERT_EQ((read (select_fd, &buf, sizeof(buf))), sizeof(buf));
- ASSERT_EQ(WatchSocket::MARKER, buf);
- // Really nothing that can be done to protect against this, but let's
- // make sure we aren't in a weird state.
- ASSERT_NO_THROW(watch->clearReady());
- // Verify the select_fd fails as socket is invalid/closed.
- EXPECT_EQ(0, selectCheck(select_fd));
- // Verify that getSelectFd() returns is still good.
- ASSERT_EQ(select_fd, watch->getSelectFd());
- }
- /// @brief Checks behavior when select_fd has been marked ready but then
- /// contents have been "corrupted" by a partial read.
- TEST(WatchSocketTest, badReadOnClear) {
- WatchSocketPtr watch;
- /// Verify that we can construct a WatchSocket.
- ASSERT_NO_THROW(watch.reset(new WatchSocket()));
- ASSERT_TRUE(watch);
- /// Verify that post-construction the state the select-fd is valid.
- int select_fd = watch->getSelectFd();
- ASSERT_NE(select_fd, WatchSocket::INVALID_SOCKET);
- /// Verify that the socket can be marked ready.
- ASSERT_NO_THROW(watch->markReady());
- EXPECT_EQ(1, selectCheck(select_fd));
- // Interfere by reading the fd. This should empty the read pipe.
- uint32_t buf = 0;
- ASSERT_EQ((read (select_fd, &buf, 1)), 1);
- ASSERT_NE(WatchSocket::MARKER, buf);
- // Really nothing that can be done to protect against this, but let's
- // make sure we aren't in a weird state.
- /// @todo maybe clear should never throw, log only
- ASSERT_THROW(watch->clearReady(), WatchSocketError);
- // Verify the select_fd does not evalute to ready.
- EXPECT_NE(1, selectCheck(select_fd));
- // Verify that getSelectFd() returns INVALID.
- ASSERT_EQ(WatchSocket::INVALID_SOCKET, watch->getSelectFd());
- // Verify that subsequent attempt to mark it fails.
- ASSERT_THROW(watch->markReady(), WatchSocketError);
- }
- } // end of anonymous namespace
|