knjrbfw 0.0.8 → 0.0.9

Sign up to get free protection for your applications and to get access to all the features.
Files changed (211) hide show
  1. data/VERSION +1 -1
  2. data/knjrbfw.gemspec +29 -8
  3. data/lib/knj/arrayext.rb +206 -153
  4. data/lib/knj/autoload/backups/ping.rb +2 -2
  5. data/lib/knj/autoload/erubis.rb +3 -3
  6. data/lib/knj/autoload/facets_dictionary.rb +2 -2
  7. data/lib/knj/autoload/gettext.rb +3 -3
  8. data/lib/knj/autoload/gtk2.rb +1 -1
  9. data/lib/knj/autoload/json.rb +14 -0
  10. data/lib/knj/autoload/magick.rb +3 -3
  11. data/lib/knj/autoload/mysql.rb +3 -3
  12. data/lib/knj/autoload/parsedate.rb +1 -1
  13. data/lib/knj/autoload/ping.rb +2 -2
  14. data/lib/knj/autoload/rexml.rb +5 -5
  15. data/lib/knj/autoload/soap.rb +1 -1
  16. data/lib/knj/autoload/sqlite3.rb +3 -3
  17. data/lib/knj/autoload/tmail.rb +3 -3
  18. data/lib/knj/autoload/xmlsimple.rb +3 -3
  19. data/lib/knj/autoload/zip.rb +3 -3
  20. data/lib/knj/autoload.rb +87 -81
  21. data/lib/knj/cmd_gen.rb +19 -19
  22. data/lib/knj/cmd_parser.rb +59 -0
  23. data/lib/knj/compiler.rb +34 -34
  24. data/lib/knj/cpufreq.rb +37 -37
  25. data/lib/knj/csv.rb +20 -0
  26. data/lib/knj/datarow.rb +406 -239
  27. data/lib/knj/datarow_custom.rb +124 -0
  28. data/lib/knj/datestamp.rb +89 -89
  29. data/lib/knj/datet.rb +550 -454
  30. data/lib/knj/db.rb +1 -0
  31. data/lib/knj/degulesider.rb +42 -44
  32. data/lib/knj/erb/erb.rb +5 -5
  33. data/lib/knj/erb/erb_cache_clean.rb +10 -10
  34. data/lib/knj/erb/erb_fcgi.rb +32 -32
  35. data/lib/knj/erb/erb_fcgi_1.9.rb +32 -32
  36. data/lib/knj/erb/erb_fcgi_jruby.rb +2 -2
  37. data/lib/knj/erb/erb_jruby.rb +5 -5
  38. data/lib/knj/erb/include.rb +243 -243
  39. data/lib/knj/errors.rb +19 -18
  40. data/lib/knj/eruby.rb +12 -10
  41. data/lib/knj/event_filemod.rb +31 -31
  42. data/lib/knj/event_handler.rb +73 -67
  43. data/lib/knj/exchangerates.rb +37 -37
  44. data/lib/knj/facebook_connect.rb +95 -18
  45. data/lib/knj/filesystem.rb +6 -6
  46. data/lib/knj/fs/drivers/filesystem.rb +12 -12
  47. data/lib/knj/fs/drivers/ftp.rb +31 -31
  48. data/lib/knj/fs/drivers/ssh.rb +26 -26
  49. data/lib/knj/fs/fs.rb +31 -31
  50. data/lib/knj/gettext_fallback.rb +15 -15
  51. data/lib/knj/gettext_threadded.rb +75 -75
  52. data/lib/knj/google_sitemap.rb +53 -53
  53. data/lib/knj/gtk2.rb +272 -272
  54. data/lib/knj/gtk2_cb.rb +80 -80
  55. data/lib/knj/gtk2_menu.rb +55 -55
  56. data/lib/knj/gtk2_statuswindow.rb +62 -62
  57. data/lib/knj/gtk2_tv.rb +58 -58
  58. data/lib/knj/hash_methods.rb +27 -36
  59. data/lib/knj/http.rb +189 -167
  60. data/lib/knj/http2.rb +259 -53
  61. data/lib/knj/image.rb +2 -2
  62. data/lib/knj/includes/appserver_cli.rb +18 -18
  63. data/lib/knj/includes/require_info.rb +15 -0
  64. data/lib/knj/ip2location.rb +20 -20
  65. data/lib/knj/ironruby-gtk2/button.rb +14 -14
  66. data/lib/knj/ironruby-gtk2/dialog.rb +42 -42
  67. data/lib/knj/ironruby-gtk2/entry.rb +4 -4
  68. data/lib/knj/ironruby-gtk2/gdk_event.rb +1 -1
  69. data/lib/knj/ironruby-gtk2/gdk_eventbutton.rb +11 -11
  70. data/lib/knj/ironruby-gtk2/gdk_pixbuf.rb +9 -9
  71. data/lib/knj/ironruby-gtk2/gladexml.rb +102 -102
  72. data/lib/knj/ironruby-gtk2/glib.rb +13 -13
  73. data/lib/knj/ironruby-gtk2/gtk2.rb +121 -121
  74. data/lib/knj/ironruby-gtk2/gtk_builder.rb +29 -29
  75. data/lib/knj/ironruby-gtk2/gtk_cellrenderertext.rb +3 -3
  76. data/lib/knj/ironruby-gtk2/gtk_combobox.rb +17 -17
  77. data/lib/knj/ironruby-gtk2/gtk_filechooserbutton.rb +3 -3
  78. data/lib/knj/ironruby-gtk2/gtk_liststore.rb +16 -16
  79. data/lib/knj/ironruby-gtk2/gtk_menu.rb +3 -3
  80. data/lib/knj/ironruby-gtk2/gtk_menuitem.rb +2 -2
  81. data/lib/knj/ironruby-gtk2/gtk_statusicon.rb +3 -3
  82. data/lib/knj/ironruby-gtk2/gtk_treeiter.rb +22 -22
  83. data/lib/knj/ironruby-gtk2/gtk_treeselection.rb +11 -11
  84. data/lib/knj/ironruby-gtk2/gtk_treeview.rb +17 -17
  85. data/lib/knj/ironruby-gtk2/gtk_treeviewcolumn.rb +30 -30
  86. data/lib/knj/ironruby-gtk2/iconsize.rb +1 -1
  87. data/lib/knj/ironruby-gtk2/image.rb +13 -13
  88. data/lib/knj/ironruby-gtk2/label.rb +17 -17
  89. data/lib/knj/ironruby-gtk2/stock.rb +2 -2
  90. data/lib/knj/ironruby-gtk2/tests/test_2.rb +14 -14
  91. data/lib/knj/ironruby-gtk2/tests/test_ironruby_window.rb +37 -37
  92. data/lib/knj/ironruby-gtk2/vbox.rb +3 -3
  93. data/lib/knj/ironruby-gtk2/window.rb +18 -18
  94. data/lib/knj/jruby-gtk2/builder/test_builder.rb +11 -11
  95. data/lib/knj/jruby-gtk2/builder.rb +25 -25
  96. data/lib/knj/jruby-gtk2/cellrenderertext.rb +10 -10
  97. data/lib/knj/jruby-gtk2/checkbutton.rb +1 -1
  98. data/lib/knj/jruby-gtk2/combobox.rb +27 -27
  99. data/lib/knj/jruby-gtk2/dialog.rb +40 -40
  100. data/lib/knj/jruby-gtk2/eventbutton.rb +19 -19
  101. data/lib/knj/jruby-gtk2/gladexml.rb +97 -97
  102. data/lib/knj/jruby-gtk2/gtk2.rb +203 -203
  103. data/lib/knj/jruby-gtk2/hbox.rb +8 -8
  104. data/lib/knj/jruby-gtk2/iconsize.rb +3 -3
  105. data/lib/knj/jruby-gtk2/image.rb +8 -8
  106. data/lib/knj/jruby-gtk2/liststore.rb +54 -54
  107. data/lib/knj/jruby-gtk2/menu.rb +35 -35
  108. data/lib/knj/jruby-gtk2/progressbar.rb +10 -10
  109. data/lib/knj/jruby-gtk2/statusicon.rb +3 -3
  110. data/lib/knj/jruby-gtk2/stock.rb +6 -6
  111. data/lib/knj/jruby-gtk2/tests/test_glade_window.rb +36 -36
  112. data/lib/knj/jruby-gtk2/tests/test_normal_window.rb +3 -3
  113. data/lib/knj/jruby-gtk2/tests/test_trayicon.rb +2 -2
  114. data/lib/knj/jruby-gtk2/treeview.rb +88 -88
  115. data/lib/knj/jruby-gtk2/vbox.rb +10 -10
  116. data/lib/knj/jruby-gtk2/window.rb +7 -7
  117. data/lib/knj/jruby_compiler.rb +12 -12
  118. data/lib/knj/knj.rb +12 -11
  119. data/lib/knj/knj_controller.rb +13 -8
  120. data/lib/knj/knjdb/drivers/mysql/knjdb_mysql.rb +499 -316
  121. data/lib/knj/knjdb/drivers/mysql/knjdb_mysql_columns.rb +127 -125
  122. data/lib/knj/knjdb/drivers/mysql/knjdb_mysql_indexes.rb +43 -25
  123. data/lib/knj/knjdb/drivers/mysql/knjdb_mysql_tables.rb +316 -241
  124. data/lib/knj/knjdb/drivers/sqlite3/knjdb_sqlite3.rb +12 -0
  125. data/lib/knj/knjdb/drivers/sqlite3/knjdb_sqlite3_columns.rb +146 -146
  126. data/lib/knj/knjdb/drivers/sqlite3/knjdb_sqlite3_indexes.rb +24 -24
  127. data/lib/knj/knjdb/drivers/sqlite3/knjdb_sqlite3_tables.rb +283 -283
  128. data/lib/knj/knjdb/libknjdb.rb +96 -52
  129. data/lib/knj/knjdb/libknjdb_java_sqlite3.rb +78 -78
  130. data/lib/knj/knjdb/libknjdb_row.rb +147 -147
  131. data/lib/knj/knjdb/libknjdb_sqlite3_ironruby.rb +61 -61
  132. data/lib/knj/knjdb/revision.rb +262 -0
  133. data/lib/knj/libqt.rb +69 -69
  134. data/lib/knj/libqt_window.rb +8 -8
  135. data/lib/knj/locales.rb +62 -52
  136. data/lib/knj/maemo/fremantle-calendar/fremantle-calendar.rb +54 -54
  137. data/lib/knj/mail.rb +73 -73
  138. data/lib/knj/mailobj.rb +82 -80
  139. data/lib/knj/mount.rb +113 -113
  140. data/lib/knj/notify.rb +11 -11
  141. data/lib/knj/nvidia_settings.rb +43 -43
  142. data/lib/knj/objects/objects_sqlhelper.rb +474 -0
  143. data/lib/knj/objects.rb +569 -793
  144. data/lib/knj/opts.rb +38 -38
  145. data/lib/knj/os.rb +176 -158
  146. data/lib/knj/php.rb +853 -683
  147. data/lib/knj/php_parser/arguments.rb +3 -3
  148. data/lib/knj/php_parser/functions.rb +95 -95
  149. data/lib/knj/php_parser/php_parser.rb +16 -16
  150. data/lib/knj/php_parser/tests/test.rb +6 -6
  151. data/lib/knj/php_parser/tests/test_function_run.rb +6 -6
  152. data/lib/knj/power_manager.rb +29 -29
  153. data/lib/knj/process.rb +457 -0
  154. data/lib/knj/process_meta.rb +400 -0
  155. data/lib/knj/rand.rb +8 -8
  156. data/lib/knj/retry.rb +69 -69
  157. data/lib/knj/rhodes/delegate.rb +414 -0
  158. data/lib/knj/rhodes/rhodes.rb +114 -40
  159. data/lib/knj/rhodes/weakref.rb +80 -0
  160. data/lib/knj/rsvgbin.rb +18 -18
  161. data/lib/knj/scripts/degulesider.rb +2 -2
  162. data/lib/knj/scripts/filesearch.rb +35 -35
  163. data/lib/knj/scripts/ip2location.rb +2 -2
  164. data/lib/knj/scripts/keepalive.rb +2 -2
  165. data/lib/knj/scripts/php_to_rb_helper.rb +376 -0
  166. data/lib/knj/scripts/process_meta_exec.rb +104 -0
  167. data/lib/knj/scripts/svn_merge.rb +21 -21
  168. data/lib/knj/scripts/upgrade_knjrbfw_checker.rb +26 -26
  169. data/lib/knj/sms.rb +52 -52
  170. data/lib/knj/sshrobot/sshrobot.rb +90 -88
  171. data/lib/knj/sshrobot.rb +1 -0
  172. data/lib/knj/strings.rb +186 -74
  173. data/lib/knj/sysuser.rb +25 -25
  174. data/lib/knj/table_writer.rb +97 -0
  175. data/lib/knj/tests/compiler/compiler_test.rb +2 -2
  176. data/lib/knj/tests/test_degulesider.rb +1 -1
  177. data/lib/knj/tests/test_http2_proxy.rb +26 -0
  178. data/lib/knj/tests/test_mount.rb +9 -9
  179. data/lib/knj/tests/test_retry.rb +17 -17
  180. data/lib/knj/thread.rb +23 -24
  181. data/lib/knj/thread2.rb +45 -45
  182. data/lib/knj/threadhandler.rb +135 -102
  183. data/lib/knj/threadpool.rb +195 -145
  184. data/lib/knj/translations.rb +128 -119
  185. data/lib/knj/unix_proc.rb +80 -80
  186. data/lib/knj/web.rb +947 -881
  187. data/lib/knj/webscripts/image.rhtml +142 -67
  188. data/lib/knj/win.rb +2 -2
  189. data/lib/knj/win_registry.rb +58 -58
  190. data/lib/knj/win_tightvnc.rb +125 -125
  191. data/lib/knj/wref.rb +104 -0
  192. data/lib/knj/x11vnc.rb +46 -46
  193. data/lib/knj/youtube.rb +33 -36
  194. data/lib/knjrbfw.rb +1 -0
  195. data/spec/cmd_parser_spec.rb +25 -0
  196. data/spec/db_spec.rb +40 -0
  197. data/spec/db_spec_encoding_test_file.txt +1 -0
  198. data/spec/http2_spec.rb +37 -0
  199. data/spec/knjrbfw_spec.rb +17 -15
  200. data/spec/php_spec.rb +69 -0
  201. data/spec/process_meta_spec.rb +150 -0
  202. data/spec/process_spec.rb +107 -0
  203. data/spec/strings_spec.rb +21 -0
  204. data/spec/web_spec.rb +16 -0
  205. metadata +42 -21
  206. data/lib/knj/autoload/json_autoload.rb +0 -7
  207. data/lib/knj/autoload/twitter.rb +0 -2
  208. data/lib/knj/ext/webrick.rb +0 -31
  209. data/lib/knj/jruby-gtk2/gtk-4.0.jar +0 -0
  210. data/lib/knj/knjdb/mysql-connector-java-5.1.13-bin.jar +0 -0
  211. data/lib/knj/knjdb/sqlitejdbc-v056.jar +0 -0
@@ -0,0 +1,457 @@
1
+ require "#{$knjpath}errors"
2
+ require "#{$knjpath}thread"
3
+
4
+ class Knj::Process
5
+ attr_reader :blocks, :blocks_send
6
+
7
+ #Constructor. Sets in, out and various other needed variables.
8
+ def initialize(args = {})
9
+ @args = args
10
+ @in = @args[:in]
11
+ @out = @args[:out]
12
+ @on_rec = @args[:on_rec]
13
+ @in_count = 0
14
+ @out_count = 0
15
+ @out_answers = {}
16
+ @out_mutex = Mutex.new
17
+ @debug = @args[:debug]
18
+ @args[:sleep_answer] = 0.001 if !@args.key?(:sleep_answer)
19
+ @thread_error = nil
20
+
21
+ #Used when this process is trying to receive block-results from the subprocess.
22
+ @blocks = {}
23
+
24
+ #Used when the other process is trying to receive block-results from this object.
25
+ @blocks_send = {}
26
+
27
+ #Else the sockets might hang when waiting for results and stuff like that... - knj.
28
+ @in.sync = true
29
+ @out.sync = true
30
+
31
+ if @args[:err]
32
+ @err_thread = Knj::Thread.new do
33
+ @args[:err].each_line do |str|
34
+ if @args[:on_err]
35
+ @args[:on_err].call(str)
36
+ else
37
+ $stderr.print "Process error: #{str}"
38
+ end
39
+ end
40
+ end
41
+ end
42
+
43
+ if @args[:listen]
44
+ require "#{$knjpath}/thread"
45
+ @listen_thread = Knj::Thread.new do
46
+ begin
47
+ self.listen
48
+ rescue Exception => e
49
+ $stderr.print "#{Knj::Errors.error_str(e)}\n\n" if @debug
50
+ @thread_error = e
51
+ end
52
+ end
53
+ end
54
+ end
55
+
56
+ def kill_listen
57
+ @listen_thread.kill if @listen_thread
58
+ end
59
+
60
+ def join
61
+ @listen_thread.join if @listen_thread
62
+ sleep 0.5
63
+ raise @thread_error if @thread_error
64
+ end
65
+
66
+ #Listens for a new incoming object.
67
+ def listen
68
+ loop do
69
+ self.listen_loop
70
+ end
71
+ end
72
+
73
+ def listen_loop
74
+ str = @in.gets("\n")
75
+ if str == nil
76
+ raise "Socket closed." if @in.closed?
77
+ sleep 0.1
78
+ return nil
79
+ end
80
+
81
+
82
+ data = str.strip.split(":")
83
+ raise "Expected length of 2 or 3 but got: '#{data.length}'.\n#{data}" if data.length != 2 and data.length != 3
84
+
85
+ raise "Invalid ID: '#{data[1]}'." if data[1].to_s.strip.length <= 0
86
+ id = data[1].to_i
87
+
88
+ raise "Invalid length: '#{data[2]}' (#{str.to_s.strip})." if data[2].to_s.strip.length <= 0
89
+ length = data[2].to_i
90
+
91
+ $stderr.print "Received ID #{id}.\n" if @debug
92
+ res = @in.read(length)
93
+
94
+ begin
95
+ obj = Marshal.load(res)
96
+ $stderr.print "Got content for '#{id}' (#{data[0]}).\n" if @debug
97
+
98
+ case data[0]
99
+ when "answer"
100
+ #raise "Already have answer for '#{id}'." if @out_answers.key?(id)
101
+ @out_answers[id] = obj
102
+ when "answer_block"
103
+ @blocks[id][:mutex].synchronize do
104
+ @blocks[id][:results] += obj
105
+ end
106
+ when "answer_block_end"
107
+ $stderr.print "Answer-block-end received!\n" if @debug
108
+ @blocks[id][:block_result] = obj
109
+ @blocks[id][:finished] = true
110
+ when "send"
111
+ Knj::Thread.new do
112
+ result_obj = Knj::Process::Resultobject.new(:process => self, :id => id, :obj => obj)
113
+
114
+ begin
115
+ @on_rec.call(result_obj)
116
+ rescue SystemExit => e
117
+ raise e
118
+ rescue Exception => e
119
+ #Error was raised - try to forward it to the server.
120
+ result_obj.answer("type" => "process_error", "class" => e.class.name, "msg" => e.message, "backtrace" => e.backtrace)
121
+ end
122
+ end
123
+ when "send_block"
124
+ result_obj = Knj::Process::Resultobject.new(:process => self, :id => id, :obj => obj)
125
+ @blocks_send[id] = {:result_obj => result_obj, :waiting_for_result => false}
126
+
127
+ @blocks_send[id][:enum] = Enumerator.new do |y|
128
+ @on_rec.call(result_obj) do |answer_block|
129
+ $stderr.print "Running enum-block for #{answer_block}\n" if @debug
130
+
131
+ break if !@blocks_send.key?(id)
132
+ y << answer_block
133
+ dobreak = false
134
+
135
+ loop do
136
+ if !@blocks_send.key?(id)
137
+ dobreak = true
138
+ break
139
+ end
140
+
141
+ break if @blocks_send[id][:waiting_for_result]
142
+ sleep 0.01
143
+ end
144
+
145
+ break if dobreak
146
+ end
147
+ end
148
+ when "send_block_res"
149
+ begin
150
+ @blocks_send[obj][:waiting_for_result] = true
151
+ res = @blocks_send[obj][:enum].next
152
+ self.answer(id, {"result" => res})
153
+ rescue StopIteration
154
+ self.answer(id, "StopIteration")
155
+ end
156
+ when "send_block_end"
157
+ if @blocks_send.key?(obj)
158
+ enum = @blocks_send[obj][:enum]
159
+ @blocks_send.delete(obj)
160
+
161
+ begin
162
+ enum.next if enum #this has to be called to stop Enumerator from blocking...
163
+ rescue StopIteration
164
+ #do nothing.
165
+ end
166
+ end
167
+
168
+ self.answer(id, "ok")
169
+ when "send_block_buffer"
170
+ buffer_use = true
171
+
172
+ Knj::Thread.new do
173
+ result_obj = Knj::Process::Resultobject.new(:process => self, :id => id, :obj => obj)
174
+ block_res = nil
175
+
176
+ begin
177
+ if buffer_use
178
+ buffer_answers = []
179
+ buffer_done = false
180
+
181
+ buffer_thread = Knj::Thread.new do
182
+ loop do
183
+ arr = buffer_answers.shift(200)
184
+
185
+ if !arr.empty?
186
+ $stderr.print "Sending: #{arr.length} results.\n" if @debug
187
+ self.answer(id, arr, "answer_block")
188
+ else
189
+ sleep 0.05
190
+ end
191
+
192
+ break if buffer_done and buffer_answers.empty?
193
+ end
194
+ end
195
+ end
196
+
197
+ begin
198
+ begin
199
+ count = 0
200
+ block_res = @on_rec.call(result_obj) do |answer_block|
201
+ if buffer_use
202
+ loop do
203
+ if buffer_answers.length > 1000
204
+ $stderr.print "Buffer is more than 1000 - sleeping and tries again in 0.05 sec.\n" if @debug
205
+ sleep 0.05
206
+ else
207
+ break
208
+ end
209
+ end
210
+ end
211
+
212
+ count += 1
213
+ if buffer_use
214
+ buffer_answers << answer_block
215
+ else
216
+ self.answer(id, [answer_block], "answer_block")
217
+ end
218
+
219
+ if count >= 100
220
+ count = 0
221
+
222
+ loop do
223
+ answer = self.send("obj" => id, "type" => "send_block_count")
224
+ $stderr.print "Answer was: #{answer}\n" if @debug
225
+
226
+ if answer >= 100
227
+ $stderr.print "More than 100 results are buffered on the other side - wait for them to be handeled before sending more.\n" if @debug
228
+ sleep 0.05
229
+ else
230
+ $stderr.print "Less than 100 results in buffer - send more.\n" if @debug
231
+ break
232
+ end
233
+ end
234
+ end
235
+ end
236
+ ensure
237
+ buffer_done = true if buffer_use
238
+ end
239
+ ensure
240
+ buffer_thread.join if buffer_use
241
+ end
242
+ rescue Exception => e
243
+ $stderr.print Knj::Errors.error_str(e) if @debug
244
+ #Error was raised - try to forward it to the server.
245
+ result_obj.answer("type" => "process_error", "class" => e.class.name, "msg" => e.message, "backtrace" => e.backtrace)
246
+ ensure
247
+ $stderr.print "Answering with block-end.\n" if @debug
248
+ self.answer(id, block_res, "answer_block_end")
249
+ end
250
+ end
251
+ when "send_block_count"
252
+ if @blocks.key?(obj)
253
+ count = @blocks[obj][:results].length
254
+ else
255
+ count = 0
256
+ end
257
+
258
+ self.answer(id, count)
259
+ else
260
+ $stderr.print "Unknown command: '#{data[0]}'."
261
+ raise "Unknown command: '#{data[0]}'."
262
+ end
263
+ rescue Exception => e
264
+ $stderr.print Knj::Errors.error_str(e) if @debug
265
+ #Error was raised - try to forward it to the server.
266
+ result_obj = Knj::Process::Resultobject.new(:process => self, :id => id, :obj => obj)
267
+ result_obj.answer("type" => "process_error", "class" => e.class.name, "msg" => e.message, "backtrace" => e.backtrace)
268
+ end
269
+ end
270
+
271
+ #Answers a send.
272
+ def answer(id, obj, type = "answer")
273
+ $stderr.print "Answering #{id} (#{obj}).\n" if @debug
274
+ str = Marshal.dump(obj)
275
+
276
+ @out_mutex.synchronize do
277
+ @out.write("#{type}:#{id}:#{str.length}\n#{str}")
278
+ end
279
+ end
280
+
281
+ #Sends a command to the client.
282
+ def send(args, &block)
283
+ args = {"obj" => args} if !args.is_a?(Hash)
284
+
285
+ my_id = nil
286
+ raise "No 'obj' was given." if !args["obj"]
287
+ str = Marshal.dump(args["obj"])
288
+
289
+ if args.key?("type")
290
+ type = args["type"]
291
+ else
292
+ type = "send"
293
+ end
294
+
295
+ raise "Invalid type: '#{type}'." if type.to_s.strip.length <= 0
296
+ args["wait_for_answer"] = true if !args.key?("wait_for_answer")
297
+
298
+ @out_mutex.synchronize do
299
+ my_id = @out_count
300
+ @out_count += 1
301
+
302
+ if block
303
+ if type == "send"
304
+ if args["buffer_use"]
305
+ type = "send_block_buffer"
306
+ @blocks[my_id] = {:block => block, :results => [], :finished => false, :buffer => args["buffer_use"], :mutex => Mutex.new}
307
+ else
308
+ type = "send_block"
309
+ end
310
+ end
311
+ end
312
+
313
+ $stderr.print "Writing #{type}:#{my_id}:#{args["obj"]} to socket.\n" if @debug
314
+ @out.write("#{type}:#{my_id}:#{str.length}\n#{str}")
315
+ end
316
+
317
+ #If block is broken it might never give us control to return anything - thats why we use ensure.
318
+ begin
319
+ if type == "send_block"
320
+ loop do
321
+ res = self.send("obj" => my_id, "type" => "send_block_res")
322
+
323
+ if res == "StopIteration"
324
+ break
325
+ elsif res.is_a?(Hash) and res.key?("result")
326
+ #do nothing.
327
+ else
328
+ raise "Unknown result: '#{res}'."
329
+ end
330
+
331
+ block.call(res["result"])
332
+ end
333
+ end
334
+ ensure
335
+ #Tell the subprocess we are done with the block (if break, exceptions or anything else like that was used).
336
+ if type == "send_block"
337
+ res = self.send("obj" => my_id, "type" => "send_block_end")
338
+ raise "Unknown result: '#{res}'." if res != "ok"
339
+ end
340
+
341
+ if args["wait_for_answer"]
342
+ #Make very, very short sleep, if the result is almost instant this will heavily optimize the speed, because :sleep_answer-argument wont be used.
343
+ sleep 0.00001
344
+ return self.read_answer(my_id)
345
+ end
346
+
347
+ return {:id => my_id}
348
+ end
349
+ end
350
+
351
+ #Returns true if an answer with a certain ID has arrived.
352
+ def has_answer?(id)
353
+ return @out_answers.key?(id)
354
+ end
355
+
356
+ #This is used to call the block with results and remove the results from the array.
357
+ def exec_block_results(id)
358
+ return nil if @blocks[id][:results].empty?
359
+
360
+ removes = []
361
+ @blocks[id][:mutex].synchronize do
362
+ results = @blocks[id][:results]
363
+ @blocks[id][:results] = []
364
+
365
+ results.each do |res|
366
+ removes << res
367
+ @blocks[id][:block].call(res)
368
+ end
369
+ end
370
+ end
371
+
372
+ #Waits for data with a certain ID and returns it when it exists.
373
+ def read_answer(id)
374
+ $stderr.print "Reading answer (#{id}).\n" if @debug
375
+ block_res = @blocks[id]
376
+ $stderr.print "Answer is block for #{id} #{block_res}\n" if @debug and block_res
377
+
378
+ loop do
379
+ if block_res
380
+ self.exec_block_results(id)
381
+ break if block_res and block_res[:finished]
382
+ else
383
+ break if @out_answers.key?(id)
384
+ end
385
+
386
+ sleep @args[:sleep_answer]
387
+ end
388
+
389
+ if block_res
390
+ self.exec_block_results(id)
391
+ @blocks.delete(id)
392
+ end
393
+
394
+ ret = @out_answers[id]
395
+ @out_answers.delete(id)
396
+
397
+ if ret.is_a?(Hash) and ret["type"] == "process_error"
398
+ $stderr.print "Process-error - begin generating error...\n" if @debug
399
+ err = RuntimeError.new(ret["msg"])
400
+ bt = []
401
+ ret["backtrace"].each do |subproc_bt|
402
+ bt << "Subprocess: #{subproc_bt}"
403
+ end
404
+
405
+ caller.each do |proc_bt|
406
+ bt << proc_bt
407
+ end
408
+
409
+ err.set_backtrace(bt)
410
+ raise err
411
+ end
412
+
413
+ $stderr.print "Returning answer (#{id}).\n" if @debug
414
+ return ret
415
+ end
416
+
417
+ #Closes everything down...
418
+ def destroy
419
+ self.kill_listen
420
+ @err_thread.kill if @err_thread
421
+ end
422
+ end
423
+
424
+ class Knj::Process::Resultobject
425
+ attr_reader :args
426
+
427
+ def initialize(args)
428
+ @args = args
429
+ @answered = false
430
+ end
431
+
432
+ #The object that was passed to the current process/socket.
433
+ def obj
434
+ return @args[:obj]
435
+ end
436
+
437
+ #Returns the process that spawned this resultobject.
438
+ def process
439
+ return @args[:process]
440
+ end
441
+
442
+ #Returns the ID this result-object should answer to.
443
+ def id
444
+ return @args[:id]
445
+ end
446
+
447
+ #Answers the call with the given object.
448
+ def answer(obj)
449
+ @answered = true
450
+ @args[:process].answer(@args[:id], obj)
451
+ end
452
+
453
+ #Returns true if this result has been answered.
454
+ def answered?
455
+ return @answered
456
+ end
457
+ end