socket.xml 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. <class name="Socket">
  2. <relationships>
  3. <association name="parent" class="Server" min="1" max="1"/>
  4. </relationships>
  5. <constructor>
  6. <parameter name="my_socket"/>
  7. <body>
  8. <![CDATA[
  9. self.socket = my_socket
  10. self.received_data = ""
  11. self.send_data = ""
  12. self.closed = False
  13. self.association_name = None
  14. ]]>
  15. </body>
  16. </constructor>
  17. <scxml initial="init">
  18. <state id="init">
  19. <transition event="set_association_name" target="../connected">
  20. <parameter name="association_name"/>
  21. <script>
  22. self.association_name = association_name
  23. </script>
  24. </transition>
  25. </state>
  26. <parallel id="connected">
  27. <state id="listening" initial="listen">
  28. <state id="listen">
  29. <onentry>
  30. <raise scope="output" port="socket_out" event="recv_socket">
  31. <parameter expr="self.socket"/>
  32. </raise>
  33. </onentry>
  34. <transition event="received_socket" port="socket_in" cond="(self.socket == socket) and (len(data) > 0)" target=".">
  35. <parameter name="socket"/>
  36. <parameter name="data"/>
  37. <script>
  38. self.received_data += data
  39. </script>
  40. <raise event="received_data"/>
  41. </transition>
  42. <transition event="received_socket" port="socket_in" cond="(self.socket == socket) and (len(data) == 0)" target="../closed">
  43. <parameter name="socket"/>
  44. <parameter name="data"/>
  45. <raise event="received_data"/>
  46. </transition>
  47. </state>
  48. <state id="closed">
  49. <onentry>
  50. <script>
  51. self.closed = True
  52. </script>
  53. </onentry>
  54. </state>
  55. </state>
  56. <state id="sending" initial="waiting_for_data">
  57. <state id="waiting_for_data">
  58. <transition cond="len(self.send_data) > 0" target="../transferring">
  59. <raise scope="output" port="socket_out" event="send_socket">
  60. <parameter expr="self.socket"/>
  61. <parameter expr="self.send_data"/>
  62. </raise>
  63. </transition>
  64. </state>
  65. <state id="transferring">
  66. <transition event="sent_socket" port="socket_in" cond="self.socket == socket" target="../waiting_for_data">
  67. <parameter name="socket"/>
  68. <parameter name="sent_bytes"/>
  69. <script>
  70. self.send_data = self.send_data[sent_bytes:]
  71. </script>
  72. </transition>
  73. </state>
  74. </state>
  75. <state id="queueing">
  76. <state id="queueing">
  77. <transition event="HTTP_input" target=".">
  78. <parameter name="data"/>
  79. <script>
  80. #post_data = "&amp;".join(["%s=%s" % (k, v) for k, v in data.iteritems()])
  81. post_data = data
  82. self.send_data += "HTTP/1.0 200 OK\r\n"
  83. self.send_data += "Content-Length: %s\r\n" % len(post_data)
  84. self.send_data += "Content-Type: %s; charset=UTF-8\r\n" % "text/plain"
  85. self.send_data += "\r\n"
  86. self.send_data += str(post_data)
  87. </script>
  88. </transition>
  89. </state>
  90. </state>
  91. <state id="parsing" initial="wait_for_header">
  92. <state id="wait_for_header">
  93. <transition cond="'\r\n\r\n' in self.received_data and self.received_data.startswith('POST')" target="../wait_for_payload">
  94. <script>
  95. header, self.received_data = self.received_data.split("\r\n\r\n", 1)
  96. header = header.lower()
  97. if "content-length" in header:
  98. _, after = header.split("content-length:", 1)
  99. after = after.split("\r\n", 1)[0]
  100. after = after.strip()
  101. self.length = int(after)
  102. else:
  103. self.length = float('inf')
  104. </script>
  105. </transition>
  106. <transition cond="self.closed and len(self.received_data) == 0" target="../closing">
  107. </transition>
  108. </state>
  109. <state id="closing">
  110. <transition after="0.1" target=".">
  111. <raise event="close"/>
  112. </transition>
  113. </state>
  114. <state id="wait_for_payload">
  115. <transition cond="len(self.received_data) >= self.length or self.closed" target="../wait_for_header">
  116. <script>
  117. if self.length == float('inf'):
  118. data = self.received_data
  119. self.received_data = ""
  120. else:
  121. data = self.received_data[:self.length]
  122. self.received_data = self.received_data[self.length:]
  123. # We support POST data only, so everything is in the data
  124. try:
  125. params = dict([p.split('=') for p in data.split('&amp;')])
  126. data = {k: urllib.unquote_plus(v) for k, v in params.iteritems()}
  127. except:
  128. print("Problem receiving data in socket")
  129. data = {}
  130. </script>
  131. <raise event="HTTP_output" scope="narrow" target="'parent'">
  132. <parameter expr="self.association_name"/>
  133. <parameter expr="data"/>
  134. </raise>
  135. </transition>
  136. </state>
  137. </state>
  138. <transition event="close" target="../close"/>
  139. </parallel>
  140. <state id="close">
  141. <onentry>
  142. <raise port="socket_out" event="close_socket">
  143. <parameter expr="self.socket"/>
  144. </raise>
  145. <raise scope="narrow" target="'parent'" event="close_socket">
  146. <parameter expr="self.association_name"/>
  147. </raise>
  148. </onentry>
  149. </state>
  150. </scxml>
  151. </class>