steam-condenser 1.3.5 → 1.3.6

Sign up to get free protection for your applications and to get access to all the features.
Files changed (161) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGELOG.md +228 -0
  3. data/CONTRIBUTING.md +59 -0
  4. data/Gemfile.lock +12 -13
  5. data/README.md +9 -4
  6. data/lib/steam-condenser/version.rb +1 -1
  7. data/lib/steam/community/alien_swarm/alien_swarm_mission.rb +10 -9
  8. data/lib/steam/community/alien_swarm/alien_swarm_stats.rb +74 -72
  9. data/lib/steam/community/css/css_map.rb +2 -2
  10. data/lib/steam/community/css/css_stats.rb +54 -56
  11. data/lib/steam/community/css/css_weapon.rb +4 -4
  12. data/lib/steam/community/game_item.rb +2 -3
  13. data/lib/steam/community/game_leaderboard.rb +22 -15
  14. data/lib/steam/community/l4d/abstract_l4d_stats.rb +64 -61
  15. data/lib/steam/community/l4d/l4d2_stats.rb +26 -21
  16. data/lib/steam/community/steam_id.rb +8 -11
  17. data/lib/steam/community/web_api.rb +19 -14
  18. data/lib/steam/packets/s2a_info2_packet.rb +16 -14
  19. data/lib/steam/packets/s2a_info_detailed_packet.rb +24 -19
  20. data/lib/steam/packets/steam_packet_factory.rb +7 -2
  21. data/lib/steam/servers/game_server.rb +3 -2
  22. data/lib/steam/servers/master_server.rb +1 -1
  23. data/lib/steam/sockets/source_socket.rb +4 -9
  24. data/pkg/steam-condenser-1.3.5.gem +0 -0
  25. data/pkg/steam-condenser-1.3.5/Gemfile +7 -0
  26. data/pkg/steam-condenser-1.3.5/Gemfile.lock +45 -0
  27. data/pkg/steam-condenser-1.3.5/LICENSE +25 -0
  28. data/pkg/steam-condenser-1.3.5/README.md +70 -0
  29. data/pkg/steam-condenser-1.3.5/Rakefile +46 -0
  30. data/pkg/steam-condenser-1.3.5/lib/core_ext/stringio.rb +91 -0
  31. data/pkg/steam-condenser-1.3.5/lib/errors/packet_format_error.rb +13 -0
  32. data/pkg/steam-condenser-1.3.5/lib/errors/rcon_ban_error.rb +22 -0
  33. data/pkg/steam-condenser-1.3.5/lib/errors/rcon_no_auth_error.rb +21 -0
  34. data/pkg/steam-condenser-1.3.5/lib/errors/steam_condenser_error.rb +26 -0
  35. data/pkg/steam-condenser-1.3.5/lib/errors/timeout_error.rb +28 -0
  36. data/pkg/steam-condenser-1.3.5/lib/errors/web_api_error.rb +57 -0
  37. data/pkg/steam-condenser-1.3.5/lib/steam-condenser.rb +18 -0
  38. data/pkg/steam-condenser-1.3.5/lib/steam-condenser/community.rb +24 -0
  39. data/pkg/steam-condenser-1.3.5/lib/steam-condenser/servers.rb +24 -0
  40. data/pkg/steam-condenser-1.3.5/lib/steam-condenser/version.rb +11 -0
  41. data/pkg/steam-condenser-1.3.5/lib/steam/community/alien_swarm/alien_swarm_mission.rb +119 -0
  42. data/pkg/steam-condenser-1.3.5/lib/steam/community/alien_swarm/alien_swarm_stats.rb +186 -0
  43. data/pkg/steam-condenser-1.3.5/lib/steam/community/alien_swarm/alien_swarm_weapon.rb +49 -0
  44. data/pkg/steam-condenser-1.3.5/lib/steam/community/app_news.rb +133 -0
  45. data/pkg/steam-condenser-1.3.5/lib/steam/community/cacheable.rb +199 -0
  46. data/pkg/steam-condenser-1.3.5/lib/steam/community/css/css_map.rb +61 -0
  47. data/pkg/steam-condenser-1.3.5/lib/steam/community/css/css_stats.rb +142 -0
  48. data/pkg/steam-condenser-1.3.5/lib/steam/community/css/css_weapon.rb +69 -0
  49. data/pkg/steam-condenser-1.3.5/lib/steam/community/defense_grid/defense_grid_stats.rb +238 -0
  50. data/pkg/steam-condenser-1.3.5/lib/steam/community/dods/dods_class.rb +90 -0
  51. data/pkg/steam-condenser-1.3.5/lib/steam/community/dods/dods_stats.rb +62 -0
  52. data/pkg/steam-condenser-1.3.5/lib/steam/community/dods/dods_weapon.rb +67 -0
  53. data/pkg/steam-condenser-1.3.5/lib/steam/community/dota2/dota2_beta_inventory.rb +30 -0
  54. data/pkg/steam-condenser-1.3.5/lib/steam/community/dota2/dota2_inventory.rb +30 -0
  55. data/pkg/steam-condenser-1.3.5/lib/steam/community/dota2/dota2_item.rb +33 -0
  56. data/pkg/steam-condenser-1.3.5/lib/steam/community/game_achievement.rb +109 -0
  57. data/pkg/steam-condenser-1.3.5/lib/steam/community/game_class.rb +22 -0
  58. data/pkg/steam-condenser-1.3.5/lib/steam/community/game_inventory.rb +171 -0
  59. data/pkg/steam-condenser-1.3.5/lib/steam/community/game_item.rb +156 -0
  60. data/pkg/steam-condenser-1.3.5/lib/steam/community/game_item_schema.rb +131 -0
  61. data/pkg/steam-condenser-1.3.5/lib/steam/community/game_leaderboard.rb +205 -0
  62. data/pkg/steam-condenser-1.3.5/lib/steam/community/game_leaderboard_entry.rb +43 -0
  63. data/pkg/steam-condenser-1.3.5/lib/steam/community/game_stats.rb +175 -0
  64. data/pkg/steam-condenser-1.3.5/lib/steam/community/game_weapon.rb +41 -0
  65. data/pkg/steam-condenser-1.3.5/lib/steam/community/l4d/abstract_l4d_stats.rb +185 -0
  66. data/pkg/steam-condenser-1.3.5/lib/steam/community/l4d/abstract_l4d_weapon.rb +56 -0
  67. data/pkg/steam-condenser-1.3.5/lib/steam/community/l4d/l4d2_map.rb +90 -0
  68. data/pkg/steam-condenser-1.3.5/lib/steam/community/l4d/l4d2_stats.rb +174 -0
  69. data/pkg/steam-condenser-1.3.5/lib/steam/community/l4d/l4d2_weapon.rb +38 -0
  70. data/pkg/steam-condenser-1.3.5/lib/steam/community/l4d/l4d_explosive.rb +34 -0
  71. data/pkg/steam-condenser-1.3.5/lib/steam/community/l4d/l4d_map.rb +65 -0
  72. data/pkg/steam-condenser-1.3.5/lib/steam/community/l4d/l4d_stats.rb +73 -0
  73. data/pkg/steam-condenser-1.3.5/lib/steam/community/l4d/l4d_weapon.rb +26 -0
  74. data/pkg/steam-condenser-1.3.5/lib/steam/community/portal2/portal2_inventory.rb +30 -0
  75. data/pkg/steam-condenser-1.3.5/lib/steam/community/portal2/portal2_item.rb +55 -0
  76. data/pkg/steam-condenser-1.3.5/lib/steam/community/portal2/portal2_stats.rb +33 -0
  77. data/pkg/steam-condenser-1.3.5/lib/steam/community/steam_game.rb +193 -0
  78. data/pkg/steam-condenser-1.3.5/lib/steam/community/steam_group.rb +129 -0
  79. data/pkg/steam-condenser-1.3.5/lib/steam/community/steam_id.rb +498 -0
  80. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_beta_inventory.rb +28 -0
  81. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_class.rb +92 -0
  82. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_class_factory.rb +40 -0
  83. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_engineer.rb +43 -0
  84. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_golden_wrench.rb +70 -0
  85. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_inventory.rb +30 -0
  86. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_item.rb +55 -0
  87. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_medic.rb +35 -0
  88. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_sniper.rb +28 -0
  89. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_spy.rb +41 -0
  90. data/pkg/steam-condenser-1.3.5/lib/steam/community/tf2/tf2_stats.rb +77 -0
  91. data/pkg/steam-condenser-1.3.5/lib/steam/community/web_api.rb +130 -0
  92. data/pkg/steam-condenser-1.3.5/lib/steam/community/xml_data.rb +27 -0
  93. data/pkg/steam-condenser-1.3.5/lib/steam/packets/a2m_get_servers_batch2_packet.rb +59 -0
  94. data/pkg/steam-condenser-1.3.5/lib/steam/packets/a2s_info_packet.rb +24 -0
  95. data/pkg/steam-condenser-1.3.5/lib/steam/packets/a2s_player_packet.rb +31 -0
  96. data/pkg/steam-condenser-1.3.5/lib/steam/packets/a2s_rules_packet.rb +32 -0
  97. data/pkg/steam-condenser-1.3.5/lib/steam/packets/a2s_serverquery_getchallenge_packet.rb +25 -0
  98. data/pkg/steam-condenser-1.3.5/lib/steam/packets/c2m_checkmd5_packet.rb +31 -0
  99. data/pkg/steam-condenser-1.3.5/lib/steam/packets/m2a_server_batch_packet.rb +52 -0
  100. data/pkg/steam-condenser-1.3.5/lib/steam/packets/m2c_isvalidmd5_packet.rb +34 -0
  101. data/pkg/steam-condenser-1.3.5/lib/steam/packets/m2s_requestrestart_packet.rb +34 -0
  102. data/pkg/steam-condenser-1.3.5/lib/steam/packets/rcon/rcon_auth_request.rb +28 -0
  103. data/pkg/steam-condenser-1.3.5/lib/steam/packets/rcon/rcon_auth_response.rb +30 -0
  104. data/pkg/steam-condenser-1.3.5/lib/steam/packets/rcon/rcon_exec_request.rb +28 -0
  105. data/pkg/steam-condenser-1.3.5/lib/steam/packets/rcon/rcon_exec_response.rb +37 -0
  106. data/pkg/steam-condenser-1.3.5/lib/steam/packets/rcon/rcon_goldsrc_request.rb +35 -0
  107. data/pkg/steam-condenser-1.3.5/lib/steam/packets/rcon/rcon_goldsrc_response.rb +34 -0
  108. data/pkg/steam-condenser-1.3.5/lib/steam/packets/rcon/rcon_packet.rb +57 -0
  109. data/pkg/steam-condenser-1.3.5/lib/steam/packets/rcon/rcon_packet_factory.rb +43 -0
  110. data/pkg/steam-condenser-1.3.5/lib/steam/packets/rcon/rcon_terminator.rb +29 -0
  111. data/pkg/steam-condenser-1.3.5/lib/steam/packets/request_with_challenge.rb +19 -0
  112. data/pkg/steam-condenser-1.3.5/lib/steam/packets/s2a_info2_packet.rb +74 -0
  113. data/pkg/steam-condenser-1.3.5/lib/steam/packets/s2a_info_base_packet.rb +25 -0
  114. data/pkg/steam-condenser-1.3.5/lib/steam/packets/s2a_info_detailed_packet.rb +59 -0
  115. data/pkg/steam-condenser-1.3.5/lib/steam/packets/s2a_logstring_packet.rb +30 -0
  116. data/pkg/steam-condenser-1.3.5/lib/steam/packets/s2a_player_packet.rb +44 -0
  117. data/pkg/steam-condenser-1.3.5/lib/steam/packets/s2a_rules_packet.rb +48 -0
  118. data/pkg/steam-condenser-1.3.5/lib/steam/packets/s2c_challenge_packet.rb +35 -0
  119. data/pkg/steam-condenser-1.3.5/lib/steam/packets/s2m_heartbeat2_packet.rb +72 -0
  120. data/pkg/steam-condenser-1.3.5/lib/steam/packets/steam_packet.rb +52 -0
  121. data/pkg/steam-condenser-1.3.5/lib/steam/packets/steam_packet_factory.rb +108 -0
  122. data/pkg/steam-condenser-1.3.5/lib/steam/servers/game_server.rb +392 -0
  123. data/pkg/steam-condenser-1.3.5/lib/steam/servers/goldsrc_server.rb +74 -0
  124. data/pkg/steam-condenser-1.3.5/lib/steam/servers/master_server.rb +188 -0
  125. data/pkg/steam-condenser-1.3.5/lib/steam/servers/server.rb +115 -0
  126. data/pkg/steam-condenser-1.3.5/lib/steam/servers/source_server.rb +117 -0
  127. data/pkg/steam-condenser-1.3.5/lib/steam/sockets/goldsrc_socket.rb +142 -0
  128. data/pkg/steam-condenser-1.3.5/lib/steam/sockets/master_server_socket.rb +34 -0
  129. data/pkg/steam-condenser-1.3.5/lib/steam/sockets/rcon_socket.rb +103 -0
  130. data/pkg/steam-condenser-1.3.5/lib/steam/sockets/source_socket.rb +78 -0
  131. data/pkg/steam-condenser-1.3.5/lib/steam/sockets/steam_socket.rb +91 -0
  132. data/pkg/steam-condenser-1.3.5/lib/steam/steam_player.rb +143 -0
  133. data/pkg/steam-condenser-1.3.5/steam-condenser.gemspec +26 -0
  134. data/pkg/steam-condenser-1.3.5/test/core_ext/test_stringio.rb +59 -0
  135. data/pkg/steam-condenser-1.3.5/test/fixtures/invalid.xml +2 -0
  136. data/pkg/steam-condenser-1.3.5/test/fixtures/sonofthor.xml +882 -0
  137. data/pkg/steam-condenser-1.3.5/test/fixtures/status_goldsrc +3 -0
  138. data/pkg/steam-condenser-1.3.5/test/fixtures/status_source +3 -0
  139. data/pkg/steam-condenser-1.3.5/test/fixtures/valve-members.xml +231 -0
  140. data/pkg/steam-condenser-1.3.5/test/helper.rb +48 -0
  141. data/pkg/steam-condenser-1.3.5/test/steam/community/test_cacheable.rb +100 -0
  142. data/pkg/steam-condenser-1.3.5/test/steam/community/test_steam_group.rb +83 -0
  143. data/pkg/steam-condenser-1.3.5/test/steam/community/test_steam_id.rb +144 -0
  144. data/pkg/steam-condenser-1.3.5/test/steam/community/test_web_api.rb +98 -0
  145. data/pkg/steam-condenser-1.3.5/test/steam/packets/test_steam_packet.rb +37 -0
  146. data/pkg/steam-condenser-1.3.5/test/steam/servers/test_game_server.rb +296 -0
  147. data/pkg/steam-condenser-1.3.5/test/steam/servers/test_goldsrc_server.rb +59 -0
  148. data/pkg/steam-condenser-1.3.5/test/steam/servers/test_master_server.rb +131 -0
  149. data/pkg/steam-condenser-1.3.5/test/steam/servers/test_server.rb +72 -0
  150. data/pkg/steam-condenser-1.3.5/test/steam/servers/test_source_server.rb +134 -0
  151. data/pkg/steam-condenser-1.3.5/test/steam/sockets/test_goldsrc_socket.rb +127 -0
  152. data/pkg/steam-condenser-1.3.5/test/steam/sockets/test_master_server_socket.rb +42 -0
  153. data/pkg/steam-condenser-1.3.5/test/steam/sockets/test_rcon_socket.rb +118 -0
  154. data/pkg/steam-condenser-1.3.5/test/steam/sockets/test_source_socket.rb +77 -0
  155. data/pkg/steam-condenser-1.3.5/test/steam/sockets/test_steam_socket.rb +97 -0
  156. data/steam-condenser.gemspec +2 -3
  157. data/test/steam/community/test_cacheable.rb +3 -0
  158. data/test/steam/community/test_web_api.rb +15 -2
  159. data/test/steam/packets/test_steam_packet.rb +1 -1
  160. data/test/steam/sockets/test_source_socket.rb +1 -1
  161. metadata +151 -27
@@ -0,0 +1,108 @@
1
+ # This code is free software; you can redistribute it and/or modify it under
2
+ # the terms of the new BSD License.
3
+ #
4
+ # Copyright (c) 2008-2011, Sebastian Staudt
5
+
6
+ require 'bzip2-ruby'
7
+ require 'zlib'
8
+
9
+ require 'errors/packet_format_error'
10
+ require 'steam/packets/s2a_info_detailed_packet'
11
+ require 'steam/packets/a2s_info_packet'
12
+ require 'steam/packets/s2a_info2_packet'
13
+ require 'steam/packets/a2s_player_packet'
14
+ require 'steam/packets/s2a_player_packet'
15
+ require 'steam/packets/a2s_rules_packet'
16
+ require 'steam/packets/s2a_rules_packet'
17
+ require 'steam/packets/a2s_serverquery_getchallenge_packet'
18
+ require 'steam/packets/s2c_challenge_packet'
19
+ require 'steam/packets/a2m_get_servers_batch2_packet'
20
+ require 'steam/packets/m2a_server_batch_packet'
21
+ require 'steam/packets/m2c_isvalidmd5_packet'
22
+ require 'steam/packets/m2s_requestrestart_packet'
23
+ require 'steam/packets/s2a_logstring_packet'
24
+ require 'steam/packets/rcon/rcon_goldsrc_response'
25
+
26
+ # This module provides functionality to handle raw packet data, including
27
+ # data split into several UDP / TCP packets and BZIP2 compressed data. It's the
28
+ # main utility to transform data bytes into packet objects.
29
+ #
30
+ # @author Sebastian Staudt
31
+ # @see SteamPacket
32
+ module SteamPacketFactory
33
+
34
+ # Creates a new packet object based on the header byte of the given raw data
35
+ #
36
+ # @param [String] raw_data The raw data of the packet
37
+ # @raise [SteamCondenserError] if the packet header is not recognized
38
+ # @return [SteamPacket] The packet object generated from the packet data
39
+ def self.packet_from_data(raw_data)
40
+ header = raw_data[0].ord
41
+ data = raw_data[1..-1]
42
+
43
+ case header
44
+ when SteamPacket::S2A_INFO_DETAILED_HEADER
45
+ return S2A_INFO_DETAILED_Packet.new(data)
46
+ when SteamPacket::A2S_INFO_HEADER
47
+ return A2S_INFO_Packet.new
48
+ when SteamPacket::S2A_INFO2_HEADER
49
+ return S2A_INFO2_Packet.new(data)
50
+ when SteamPacket::A2S_PLAYER_HEADER
51
+ return A2S_PLAYER_Packet.new
52
+ when SteamPacket::S2A_PLAYER_HEADER
53
+ return S2A_PLAYER_Packet.new(data)
54
+ when SteamPacket::A2S_RULES_HEADER
55
+ return A2S_RULES_Packet
56
+ when SteamPacket::S2A_RULES_HEADER
57
+ return S2A_RULES_Packet.new(data)
58
+ when SteamPacket::A2S_SERVERQUERY_GETCHALLENGE_HEADER
59
+ return A2S_SERVERQUERY_GETCHALLENGE_Packet.new
60
+ when SteamPacket::S2C_CHALLENGE_HEADER
61
+ return S2C_CHALLENGE_Packet.new(data)
62
+ when SteamPacket::A2M_GET_SERVERS_BATCH2_HEADER
63
+ return A2M_GET_SERVERS_BATCH2_Packet.new(data)
64
+ when SteamPacket::M2A_SERVER_BATCH_HEADER
65
+ return M2A_SERVER_BATCH_Packet.new(data)
66
+ when SteamPacket::M2C_ISVALIDMD5_HEADER
67
+ return M2C_ISVALIDMD5_Packet.new(data)
68
+ when SteamPacket::M2S_REQUESTRESTART_HEADER
69
+ return M2S_REQUESTRESTART_Packet.new(data)
70
+ when SteamPacket::RCON_GOLDSRC_CHALLENGE_HEADER,
71
+ SteamPacket::RCON_GOLDSRC_NO_CHALLENGE_HEADER,
72
+ SteamPacket::RCON_GOLDSRC_RESPONSE_HEADER
73
+ return RCONGoldSrcResponse.new(data)
74
+ when SteamPacket::S2A_LOGSTRING_HEADER
75
+ return S2A_LOGSTRING_Packet.new(data)
76
+ else
77
+ raise PacketFormatError, "Unknown packet with header 0x#{header.to_s(16)} received."
78
+ end
79
+ end
80
+
81
+ # Reassembles the data of a split and/or compressed packet into a single
82
+ # packet object
83
+ #
84
+ # @param [Array<String>] split_packets An array of packet data
85
+ # @param [Boolean] is_compressed whether the data of this packet is
86
+ # compressed
87
+ # @param [Fixnum] packet_checksum The CRC32 checksum of the decompressed
88
+ # packet data
89
+ # @raise [SteamCondenserError] if the bz2 gem is not installed
90
+ # @raise [PacketFormatError] if the calculated CRC32 checksum does not match
91
+ # the expected value
92
+ # @return [SteamPacket] The reassembled packet
93
+ # @see packet_from_data
94
+ def self.reassemble_packet(split_packets, is_compressed = false, packet_checksum = 0)
95
+ packet_data = split_packets.join ''
96
+
97
+ if is_compressed
98
+ packet_data = Bzip2.decompress packet_data
99
+
100
+ unless Zlib.crc32(packet_data) == packet_checksum
101
+ raise PacketFormatError, 'CRC32 checksum mismatch of uncompressed packet data.'
102
+ end
103
+ end
104
+
105
+ packet_from_data packet_data[4..-1]
106
+ end
107
+
108
+ end
@@ -0,0 +1,392 @@
1
+ # This code is free software; you can redistribute it and/or modify it under
2
+ # the terms of the new BSD License.
3
+ #
4
+ # Copyright (c) 2008-2013, Sebastian Staudt
5
+
6
+ require 'errors/steam_condenser_error'
7
+ require 'errors/timeout_error'
8
+ require 'steam/steam_player'
9
+ require 'steam/packets/a2s_info_packet'
10
+ require 'steam/packets/a2s_player_packet'
11
+ require 'steam/packets/a2s_rules_packet'
12
+ require 'steam/packets/a2s_serverquery_getchallenge_packet'
13
+ require 'steam/packets/s2a_info_base_packet'
14
+ require 'steam/packets/s2a_player_packet'
15
+ require 'steam/packets/s2a_rules_packet'
16
+ require 'steam/packets/s2c_challenge_packet'
17
+ require 'steam/servers/server'
18
+
19
+ # This module is included by classes representing different game server
20
+ # implementations and provides the basic functionality to communicate with
21
+ # them using the common query protocol
22
+ #
23
+ # @author Sebastian Staudt
24
+ module GameServer
25
+
26
+ include Server
27
+
28
+ # Parses the player attribute names supplied by `rcon status`
29
+ #
30
+ # @param [String] status_header The header line provided by `rcon status`
31
+ # @return [Array<Symbol>] Split player attribute names
32
+ # @see .split_player_status
33
+ def self.player_status_attributes(status_header)
34
+ status_header.split.map do |attribute|
35
+ case attribute
36
+ when 'connected'
37
+ :time
38
+ when 'frag'
39
+ :score
40
+ else
41
+ attribute.to_sym
42
+ end
43
+ end
44
+ end
45
+
46
+ # Splits the player status obtained with `rcon status`
47
+ #
48
+ # @param [Array<Symbol>] attributes The attribute names
49
+ # @param [String] player_status The status line of a single player
50
+ # @return [Hash<Symbol, String>] The attributes with the corresponding values
51
+ # for this player
52
+ # @see .player_status_attributes
53
+ def self.split_player_status(attributes, player_status)
54
+ player_status.sub! /^\d+ +/, '' if attributes.first != :userid
55
+
56
+ first_quote = player_status.index '"'
57
+ last_quote = player_status.rindex '"'
58
+ data = [
59
+ player_status[0, first_quote],
60
+ player_status[first_quote + 1..last_quote - 1],
61
+ player_status[last_quote + 1..-1]
62
+ ]
63
+ data = [ data[0].split, data[1], data[2].split ]
64
+ data.flatten!
65
+
66
+ if attributes.size > data.size && attributes.include?(:state)
67
+ data.insert 3, nil, nil, nil
68
+ elsif attributes.size < data.size
69
+ data.delete_at 1
70
+ end
71
+
72
+ player_data = {}
73
+ data.each_index do |i|
74
+ player_data[attributes[i]] = data[i]
75
+ end
76
+
77
+ player_data
78
+ end
79
+
80
+ # Creates a new instance of a game server object
81
+ #
82
+ # @param [String] address Either an IP address, a DNS name or one of them
83
+ # combined with the port number. If a port number is given, e.g.
84
+ # 'server.example.com:27016' it will override the second argument.
85
+ # @param [Fixnum] port The port the server is listening on
86
+ # @raise [SteamCondenserError] if an host name cannot be resolved
87
+ def initialize(address, port = 27015)
88
+ super
89
+
90
+ @rcon_authenticated = false
91
+ end
92
+
93
+ # Returns the last measured response time of this server
94
+ #
95
+ # If the latency hasn't been measured yet, it is done when calling this
96
+ # method for the first time.
97
+ #
98
+ # If this information is vital to you, be sure to call {#update_ping}
99
+ # regularly to stay up-to-date.
100
+ #
101
+ # @return [Fixnum] The latency of this server in milliseconds
102
+ # @see #update_ping
103
+ def ping
104
+ update_ping if @ping.nil?
105
+ @ping
106
+ end
107
+
108
+ # Returns a list of players currently playing on this server
109
+ #
110
+ # If the players haven't been fetched yet, it is done when calling this
111
+ # method for the first time.
112
+ #
113
+ # As the players and their scores change quite often be sure to update this
114
+ # list regularly by calling {#update_players} if you rely on this
115
+ # information.
116
+ #
117
+ # @param [String] rcon_password The RCON password of this server may be
118
+ # provided to gather more detailed information on the players, like
119
+ # STEAM_IDs.
120
+ # @return [Hash] The players on this server
121
+ # @see update_players
122
+ def players(rcon_password = nil)
123
+ update_players(rcon_password) if @player_hash.nil?
124
+ @player_hash
125
+ end
126
+
127
+ # Authenticates the connection for RCON communication with the server
128
+ #
129
+ # @abstract Must be be implemented by including classes to handle the
130
+ # authentication
131
+ # @param [String] password The RCON password of the server
132
+ # @return [Boolean] whether the authentication was successful
133
+ # @see #rcon_exec
134
+ def rcon_auth(password)
135
+ raise NotImplementedError
136
+ end
137
+
138
+ # Returns whether the RCON connection to this server is already authenticated
139
+ #
140
+ # @return [Boolean] `true` if the RCON connection is authenticated
141
+ # @see #rcon_auth
142
+ def rcon_authenticated?
143
+ @rcon_authenticated
144
+ end
145
+
146
+ # Remotely executes a command on the server via RCON
147
+ #
148
+ # @abstract Must be be implemented by including classes to handle the command
149
+ # execution
150
+ # @param [String] command The command to execute on the server
151
+ # @return [String] The output of the executed command
152
+ # @see #rcon_auth
153
+ def rcon_exec(command)
154
+ raise NotImplementedError
155
+ end
156
+
157
+ # Returns the settings applied on the server. These settings are also called
158
+ # rules.
159
+ #
160
+ # If the rules haven't been fetched yet, it is done when calling this method
161
+ # for the first time.
162
+ #
163
+ # As the rules usually don't change often, there's almost no need to update
164
+ # this hash. But if you need to, you can achieve this by calling
165
+ # {#update_rules}.
166
+ #
167
+ # @return [Hash<String, String>] The currently active server rules
168
+ # @see #update_rules
169
+ def rules
170
+ update_rules if @rules_hash.nil?
171
+ @rules_hash
172
+ end
173
+
174
+ # Returns a hash with basic information on the server.
175
+ #
176
+ # If the server information haven't been fetched yet, it is done when
177
+ # calling this method for the first time.
178
+ #
179
+ # The server information usually only changes on map change and when players
180
+ # join or leave. As the latter changes can be monitored by calling
181
+ # {#update_players}, there's no need to call {#update_server_info} very
182
+ # often.
183
+ #
184
+ # @return [Hash] Server attributes with their values
185
+ # @see #update_server_info
186
+ def server_info
187
+ update_server_info if @info_hash.nil?
188
+ @info_hash
189
+ end
190
+
191
+ # Sends the specified request to the server and handles the returned response
192
+ #
193
+ # Depending on the given request type this will fill the various data
194
+ # attributes of the server object.
195
+ #
196
+ # @param [Symbol] request_type The type of request to send to the server
197
+ # @param [Boolean] repeat_on_failure Whether the request should be repeated,
198
+ # if the replied packet isn't expected. This is useful to handle
199
+ # missing challenge numbers, which will be automatically filled in,
200
+ # although not requested explicitly.
201
+ # @raise [SteamCondenserError] if either the request type or the response
202
+ # packet is not known
203
+ def handle_response_for_request(request_type, repeat_on_failure = true)
204
+ case request_type
205
+ when :challenge then
206
+ request_packet = A2S_PLAYER_Packet.new
207
+ expected_response = S2C_CHALLENGE_Packet
208
+ when :info then
209
+ request_packet = A2S_INFO_Packet.new
210
+ expected_response = S2A_INFO_BasePacket
211
+ when :players then
212
+ request_packet = A2S_PLAYER_Packet.new(@challenge_number)
213
+ expected_response = S2A_PLAYER_Packet
214
+ when :rules then
215
+ request_packet = A2S_RULES_Packet.new(@challenge_number)
216
+ expected_response = S2A_RULES_Packet
217
+ else
218
+ raise SteamCondenserError, 'Called with wrong request type.'
219
+ end
220
+
221
+ send_request request_packet
222
+ response_packet = reply
223
+
224
+ if response_packet.kind_of? S2A_INFO_BasePacket
225
+ @info_hash = response_packet.info
226
+ elsif response_packet.kind_of? S2A_PLAYER_Packet
227
+ @player_hash = response_packet.player_hash
228
+ elsif response_packet.kind_of? S2A_RULES_Packet
229
+ @rules_hash = response_packet.rules_hash
230
+ elsif response_packet.kind_of? S2C_CHALLENGE_Packet
231
+ @challenge_number = response_packet.challenge_number
232
+ else
233
+ raise SteamCondenserError, "Response of type #{response_packet.class} cannot be handled by this method."
234
+ end
235
+
236
+ unless response_packet.kind_of? expected_response
237
+ puts "Expected #{expected_response}, got #{response_packet.class}." if $DEBUG
238
+ handle_response_for_request(request_type, false) if repeat_on_failure
239
+ end
240
+ end
241
+
242
+ # Initializes this server object with basic information
243
+ #
244
+ # @see #update_challenge_number
245
+ # @see #update_ping
246
+ # @see #update_server_info
247
+ def init
248
+ update_ping
249
+ update_server_info
250
+ update_challenge_number
251
+ end
252
+
253
+ # Sends a A2S_PLAYERS request to the server and updates the players' data for
254
+ # this server
255
+ #
256
+ # As the players and their scores change quite often be sure to update this
257
+ # list regularly by calling this method if you rely on this
258
+ # information.
259
+ #
260
+ # @param [String] rcon_password The RCON password of this server may be
261
+ # provided to gather more detailed information on the players, like
262
+ # STEAM_IDs.
263
+ # @see #handle_response_for_request
264
+ # @see #players
265
+ def update_players(rcon_password = nil)
266
+ handle_response_for_request :players
267
+
268
+ unless @rcon_authenticated
269
+ return if rcon_password.nil?
270
+ rcon_auth rcon_password
271
+ end
272
+
273
+ players = rcon_exec('status').lines.select do |line|
274
+ line.start_with?('#') && line != "#end\n"
275
+ end.map do |line|
276
+ line[1..-1].strip
277
+ end
278
+ attributes = GameServer.player_status_attributes players.shift
279
+
280
+ players.each do |player|
281
+ player_data = GameServer.split_player_status(attributes, player)
282
+ if @player_hash.key? player_data[:name]
283
+ @player_hash[player_data[:name]].add_info player_data
284
+ end
285
+ end
286
+ end
287
+
288
+ # Sends a A2S_RULES request to the server and updates the rules of this
289
+ # server
290
+ #
291
+ # As the rules usually don't change often, there's almost no need to update
292
+ # this hash. But if you need to, you can achieve this by calling this method.
293
+ #
294
+ # @see #handle_response_for_request
295
+ # @see #rules
296
+ def update_rules
297
+ handle_response_for_request :rules
298
+ end
299
+
300
+ # Sends a A2S_INFO request to the server and updates this server's basic
301
+ # information
302
+ #
303
+ # The server information usually only changes on map change and when players
304
+ # join or leave. As the latter changes can be monitored by calling
305
+ # {#update_players}, there's no need to call this method very often.
306
+ #
307
+ # @see #handle_response_for_request
308
+ # @see #server_info
309
+ def update_server_info
310
+ handle_response_for_request :info
311
+ end
312
+
313
+ # Sends a A2S_SERVERQUERY_GETCHALLENGE request to the server and updates the
314
+ # challenge number used to communicate with this server
315
+ #
316
+ # There's usually no need to call this method explicitly, because
317
+ # {#handle_response_for_request} will automatically get the challenge number
318
+ # when the server assigns a new one.
319
+ #
320
+ # @see #handle_response_for_request
321
+ # @see #init
322
+ def update_challenge_number
323
+ handle_response_for_request :challenge
324
+ end
325
+
326
+ # Sends a A2S_INFO request to the server and measures the time needed for the
327
+ # reply
328
+ #
329
+ # If this information is vital to you, be sure to call this method regularly
330
+ # to stay up-to-date.
331
+ #
332
+ # @return [Fixnum] The latency of this server in milliseconds
333
+ # @see #ping
334
+ def update_ping
335
+ send_request A2S_INFO_Packet.new
336
+ start_time = Time.now
337
+ reply
338
+ end_time = Time.now
339
+ @ping = (end_time - start_time) * 1000
340
+ end
341
+
342
+ # Returns a human-readable text representation of the server
343
+ #
344
+ # @return [String] Available information about the server in a human-readable
345
+ # format
346
+ def to_s
347
+ return_string = ''
348
+
349
+ return_string << "Ping: #@ping\n"
350
+ return_string << "Challenge number: #@challenge_number\n"
351
+
352
+ unless @info_hash.nil?
353
+ return_string << "Info:\n"
354
+ @info_hash.each do |key, value|
355
+ return_string << " #{key}: #{value.inspect}\n"
356
+ end
357
+ end
358
+
359
+ unless @player_hash.nil?
360
+ return_string << "Players:\n"
361
+ @player_hash.each_value do |player|
362
+ return_string << " #{player}\n"
363
+ end
364
+ end
365
+
366
+ unless @rules_hash.nil?
367
+ return_string << "Rules:\n"
368
+ @rules_hash.each do |key, value|
369
+ return_string << " #{key}: #{value}\n"
370
+ end
371
+ end
372
+
373
+ return_string
374
+ end
375
+
376
+ protected
377
+
378
+ # Receives a response from the server
379
+ #
380
+ # @return [SteamPacket] The response packet replied by the server
381
+ def reply
382
+ @socket.reply
383
+ end
384
+
385
+ # Sends a request packet to the server
386
+ #
387
+ # @param [SteamPacket] packet The request packet to send to the server
388
+ def send_request(packet)
389
+ @socket.send packet
390
+ end
391
+
392
+ end