RemoteCommandEvent.h 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) 2013 GarageGames, LLC
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a copy
  5. // of this software and associated documentation files (the "Software"), to
  6. // deal in the Software without restriction, including without limitation the
  7. // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  8. // sell copies of the Software, and to permit persons to whom the Software is
  9. // furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  20. // IN THE SOFTWARE.
  21. //-----------------------------------------------------------------------------
  22. #ifndef _H_REMOTECOMMANDEVENT
  23. #define _H_REMOTECOMMANDEVENT
  24. class RemoteCommandEvent : public NetEvent
  25. {
  26. public:
  27. enum {
  28. MaxRemoteCommandArgs = 20,
  29. CommandArgsBits = 5
  30. };
  31. private:
  32. S32 mArgc;
  33. char *mArgv[MaxRemoteCommandArgs + 1];
  34. NetStringHandle mTagv[MaxRemoteCommandArgs + 1];
  35. static char mBuf[1024];
  36. public:
  37. RemoteCommandEvent(S32 argc=0, const char **argv=NULL, NetConnection *conn = NULL)
  38. {
  39. mArgc = argc;
  40. for(S32 i = 0; i < argc; i++)
  41. {
  42. if(argv[i][0] == StringTagPrefixByte)
  43. {
  44. char buffer[256];
  45. mTagv[i+1] = NetStringHandle(dAtoi(argv[i]+1));
  46. if(conn)
  47. {
  48. dSprintf(buffer + 1, sizeof(buffer) - 1, "%d", conn->getNetSendId(mTagv[i+1]));
  49. buffer[0] = StringTagPrefixByte;
  50. mArgv[i+1] = dStrdup(buffer);
  51. }
  52. }
  53. else
  54. mArgv[i+1] = dStrdup(argv[i]);
  55. }
  56. }
  57. #ifdef TORQUE_DEBUG_NET
  58. const char *getDebugName()
  59. {
  60. static char buffer[256];
  61. dSprintf(buffer, sizeof(buffer), "%s [%s]", getClassName(), gNetStringTable->lookupString(dAtoi(mArgv[1] + 1)) );
  62. return buffer;
  63. }
  64. #endif
  65. ~RemoteCommandEvent()
  66. {
  67. for(S32 i = 0; i < mArgc; i++)
  68. dFree(mArgv[i+1]);
  69. }
  70. virtual void pack(NetConnection* conn, BitStream *bstream)
  71. {
  72. bstream->writeInt(mArgc, CommandArgsBits);
  73. // write it out reversed... why?
  74. // automatic string substitution with later arguments -
  75. // handled automatically by the system.
  76. for(S32 i = 0; i < mArgc; i++)
  77. conn->packString(bstream, mArgv[i+1]);
  78. }
  79. virtual void write(NetConnection* conn, BitStream *bstream)
  80. {
  81. pack(conn, bstream);
  82. }
  83. virtual void unpack(NetConnection* conn, BitStream *bstream)
  84. {
  85. mArgc = bstream->readInt(CommandArgsBits);
  86. // read it out backwards
  87. for(S32 i = 0; i < mArgc; i++)
  88. {
  89. conn->unpackString(bstream, mBuf);
  90. mArgv[i+1] = dStrdup(mBuf);
  91. }
  92. }
  93. virtual void process(NetConnection *conn)
  94. {
  95. static char idBuf[10];
  96. // de-tag the command name
  97. for(S32 i = mArgc - 1; i >= 0; i--)
  98. {
  99. char *arg = mArgv[i+1];
  100. if(*arg == StringTagPrefixByte)
  101. {
  102. // it's a tag:
  103. U32 localTag = dAtoi(arg + 1);
  104. NetStringHandle tag = conn->translateRemoteStringId(localTag);
  105. NetStringTable::expandString( tag,
  106. mBuf,
  107. sizeof(mBuf),
  108. (mArgc - 1) - i,
  109. (const char**)(mArgv + i + 2) );
  110. dFree(mArgv[i+1]);
  111. mArgv[i+1] = dStrdup(mBuf);
  112. }
  113. }
  114. const char *rmtCommandName = dStrchr(mArgv[1], ' ') + 1;
  115. if(conn->isConnectionToServer())
  116. {
  117. dStrcpy(mBuf, "clientCmd");
  118. dStrcat(mBuf, rmtCommandName);
  119. char *temp = mArgv[1];
  120. mArgv[1] = mBuf;
  121. Con::execute(mArgc, (const char **) mArgv+1);
  122. mArgv[1] = temp;
  123. }
  124. else
  125. {
  126. dStrcpy(mBuf, "serverCmd");
  127. dStrcat(mBuf, rmtCommandName);
  128. char *temp = mArgv[1];
  129. dSprintf(idBuf, sizeof(idBuf), "%d", conn->getId());
  130. mArgv[0] = mBuf;
  131. mArgv[1] = idBuf;
  132. Con::execute(mArgc+1, (const char **) mArgv);
  133. mArgv[1] = temp;
  134. }
  135. }
  136. DECLARE_CONOBJECT(RemoteCommandEvent);
  137. };
  138. #endif