/src/man/include/io/SocketInProvider.h

https://github.com/nterman/nbites · C Header · 157 lines · 111 code · 36 blank · 10 comment · 13 complexity · 2cf913b10993864e1710369af122b843 MD5 · raw file

  1. /**
  2. * @class SocketInProvider
  3. *
  4. * Opens a client-type socket that is made to connect to the
  5. * specified address and port
  6. *
  7. * @author Octavian Neamtu
  8. */
  9. #pragma once
  10. #include <iostream>
  11. #include <fcntl.h>
  12. #include <string>
  13. #include <aio.h>
  14. #include <cstring>
  15. #include <pthread.h>
  16. #include "Socket.h"
  17. #include "IOExceptions.h"
  18. #include "InProvider.h"
  19. namespace common {
  20. namespace io {
  21. class SocketInProvider: public InProvider {
  22. public:
  23. SocketInProvider(long address, unsigned short port) :
  24. port(port),
  25. address(address),
  26. is_open(false),
  27. file_descriptor(-1) {
  28. //zeroes the aio control_block
  29. memset(&control_block, 0, sizeof(control_block));
  30. }
  31. virtual ~SocketInProvider() {
  32. closeChannel();
  33. }
  34. virtual void closeChannel() {
  35. is_open = false;
  36. close(file_descriptor);
  37. }
  38. virtual std::string debugInfo() const {
  39. struct in_addr addr;
  40. addr.s_addr = address;
  41. return "SocketIn client connecting to " + std::string(inet_ntoa(addr));
  42. }
  43. void waitForReadToFinish() const {
  44. const struct aiocb* cblist[] = { &control_block };
  45. struct timespec timeout;
  46. timeout.tv_nsec = 0;
  47. timeout.tv_sec = 5;
  48. int result = aio_suspend(cblist, 1, &timeout);
  49. if (result != 0) {
  50. throw_errno(errno);
  51. }
  52. }
  53. virtual bool isOfTypeStreaming() const { return true; }
  54. virtual bool reachedEnd() const { return false; }
  55. bool rewind(long int) const { return false; }
  56. virtual void readCharBuffer(char* buffer, uint32_t size) const
  57. throw (aio_read_exception) {
  58. aioReadCharBuffer(buffer, size);
  59. }
  60. virtual void aioReadCharBuffer(char* buffer, uint32_t size) const
  61. throw (aio_read_exception) {
  62. if (!opened()) {
  63. throw aio_read_exception(aio_read_exception::NOT_OPEN);
  64. }
  65. if (readInProgress()) {
  66. throw aio_read_exception(aio_read_exception::IN_PROGRESS);
  67. }
  68. enqueBuffer(buffer, size);
  69. }
  70. virtual void enqueBuffer(char* buffer, uint32_t size) const
  71. throw (aio_read_exception) {
  72. control_block.aio_fildes = file_descriptor;
  73. control_block.aio_buf = buffer;
  74. control_block.aio_nbytes = size;
  75. int result = aio_read(&control_block);
  76. if (result == -1) {
  77. throw aio_read_exception(aio_read_exception::ENQUE, errno);
  78. }
  79. }
  80. virtual uint32_t bytesRead() const throw (aio_read_exception) {
  81. if (readInProgress()) {
  82. throw aio_read_exception(aio_read_exception::IN_PROGRESS);
  83. }
  84. if (aio_error(&control_block) != 0) {
  85. throw aio_read_exception(aio_read_exception::READ, errno);
  86. }
  87. return aio_return(&control_block);
  88. }
  89. virtual bool readInProgress() const {
  90. return aio_error(&control_block) == EINPROGRESS;
  91. }
  92. void openCommunicationChannel() throw (socket_exception) {
  93. if (is_open) {
  94. return;
  95. }
  96. file_descriptor = tcp::createSocket();
  97. tcp::connectSocket(file_descriptor, address, port);
  98. is_open = true;
  99. }
  100. //blocking!
  101. virtual void peekAt(char* buffer, uint32_t size) const
  102. throw (aio_read_exception) {
  103. if (!opened()) {
  104. throw aio_read_exception(aio_read_exception::NOT_OPEN);
  105. }
  106. if (recv(file_descriptor, buffer, size, MSG_PEEK) < 0) {
  107. throw aio_read_exception(aio_read_exception::READ, errno);
  108. }
  109. }
  110. virtual bool opened() const {
  111. return is_open;
  112. }
  113. private:
  114. unsigned short port;
  115. long address;
  116. bool is_open;
  117. mutable aiocb control_block;
  118. int file_descriptor;
  119. };
  120. }
  121. }