discourse-systray 0.1.2 → 0.1.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/lib/discourse_systray/systray.rb +440 -181
- metadata +2 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 52f16a25286153255ec00c850f08e74cd2b704903d8adf850705f17aa6898fdb
|
4
|
+
data.tar.gz: 4f708705f6669dff35a737ff328917c3b603b7cc7f87c0c1d3a7602ee0157852
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: ffaf161d05e11185dde22eb1f4a8c61dd42fa14ffcc72738391e5a8f54c68523f225d8442bd83f49bdec36c15e7237bf58eb351811781f845bedd15f92592781
|
7
|
+
data.tar.gz: 5cf888a341f139472fc4858731310d42e4586910f19992643c015e1be1f95ce20140d9d37ca0b8644f8fd2afa8d363b185cc42509d1a41f96293ae338f3dfa3a
|
@@ -74,19 +74,73 @@ module ::DiscourseSystray
|
|
74
74
|
rescue JSON::ParserError
|
75
75
|
{}
|
76
76
|
end
|
77
|
-
BUFFER_SIZE =
|
77
|
+
BUFFER_SIZE = 1000
|
78
|
+
BUFFER_TRIM_INTERVAL = 30 # seconds
|
78
79
|
|
79
80
|
def initialize
|
81
|
+
puts "DEBUG: Initializing DiscourseSystray" if OPTIONS[:debug]
|
82
|
+
|
80
83
|
@discourse_path = self.class.load_or_prompt_config unless OPTIONS[:attach]
|
84
|
+
puts "DEBUG: Discourse path: #{@discourse_path}" if OPTIONS[:debug]
|
85
|
+
|
81
86
|
@running = false
|
82
87
|
@ember_output = []
|
83
88
|
@unicorn_output = []
|
84
89
|
@processes = {}
|
85
90
|
@ember_running = false
|
86
91
|
@unicorn_running = false
|
87
|
-
@ember_line_count = 0
|
88
|
-
@unicorn_line_count = 0
|
89
92
|
@status_window = nil
|
93
|
+
@buffer_trim_timer = nil
|
94
|
+
|
95
|
+
# Initialize pipe queue for background processing
|
96
|
+
initialize_pipe_queue
|
97
|
+
|
98
|
+
# Add initial welcome message to buffers with timestamp
|
99
|
+
timestamp = Time.now.strftime("%H:%M:%S")
|
100
|
+
@ember_output << "#{timestamp} - Discourse Ember CLI Log\n"
|
101
|
+
@ember_output << "Start Discourse to see Ember CLI logs here.\n"
|
102
|
+
@ember_output << "\n"
|
103
|
+
|
104
|
+
@unicorn_output << "#{timestamp} - Discourse Unicorn Log\n"
|
105
|
+
@unicorn_output << "Start Discourse to see Unicorn logs here.\n"
|
106
|
+
@unicorn_output << "\n"
|
107
|
+
|
108
|
+
# Add a visual separator
|
109
|
+
@ember_output << "=" * 50 + "\n"
|
110
|
+
@unicorn_output << "=" * 50 + "\n"
|
111
|
+
|
112
|
+
puts "DEBUG: Added initial data to buffers" if OPTIONS[:debug]
|
113
|
+
puts "DEBUG: ember_output size: #{@ember_output.size}" if OPTIONS[:debug]
|
114
|
+
puts "DEBUG: unicorn_output size: #{@unicorn_output.size}" if OPTIONS[:debug]
|
115
|
+
|
116
|
+
# Set up periodic buffer trimming
|
117
|
+
setup_buffer_trim_timer unless OPTIONS[:attach]
|
118
|
+
|
119
|
+
puts "DEBUG: Initialized DiscourseSystray with path: #{@discourse_path}" if OPTIONS[:debug]
|
120
|
+
end
|
121
|
+
|
122
|
+
def setup_buffer_trim_timer
|
123
|
+
@buffer_trim_timer = GLib::Timeout.add_seconds(BUFFER_TRIM_INTERVAL) do
|
124
|
+
trim_buffers
|
125
|
+
true # Keep the timer running
|
126
|
+
end
|
127
|
+
end
|
128
|
+
|
129
|
+
def trim_buffers
|
130
|
+
# Trim buffers if they exceed the buffer size
|
131
|
+
if @ember_output.size > BUFFER_SIZE
|
132
|
+
excess = @ember_output.size - BUFFER_SIZE
|
133
|
+
@ember_output.shift(excess)
|
134
|
+
@ember_line_count = [@ember_line_count - excess, 0].max
|
135
|
+
end
|
136
|
+
|
137
|
+
if @unicorn_output.size > BUFFER_SIZE
|
138
|
+
excess = @unicorn_output.size - BUFFER_SIZE
|
139
|
+
@unicorn_output.shift(excess)
|
140
|
+
@unicorn_line_count = [@unicorn_line_count - excess, 0].max
|
141
|
+
end
|
142
|
+
|
143
|
+
true
|
90
144
|
end
|
91
145
|
|
92
146
|
def init_systray
|
@@ -182,6 +236,27 @@ module ::DiscourseSystray
|
|
182
236
|
end
|
183
237
|
end
|
184
238
|
@view_timeouts&.clear
|
239
|
+
|
240
|
+
# Remove buffer trim timer if it exists
|
241
|
+
if @buffer_trim_timer
|
242
|
+
begin
|
243
|
+
GLib::Source.remove(@buffer_trim_timer)
|
244
|
+
@buffer_trim_timer = nil
|
245
|
+
rescue StandardError => e
|
246
|
+
puts "Error removing buffer trim timer: #{e}" if OPTIONS[:debug]
|
247
|
+
end
|
248
|
+
end
|
249
|
+
|
250
|
+
# Stop pipe thread
|
251
|
+
if @pipe_thread
|
252
|
+
begin
|
253
|
+
@pipe_queue.push(:exit) if @pipe_queue
|
254
|
+
@pipe_thread.join(2) # Wait up to 2 seconds
|
255
|
+
@pipe_thread.kill if @pipe_thread.alive?
|
256
|
+
rescue StandardError => e
|
257
|
+
puts "Error stopping pipe thread: #{e}" if OPTIONS[:debug]
|
258
|
+
end
|
259
|
+
end
|
185
260
|
|
186
261
|
# Then stop processes
|
187
262
|
@processes.each do |name, process|
|
@@ -207,31 +282,90 @@ module ::DiscourseSystray
|
|
207
282
|
end
|
208
283
|
|
209
284
|
def start_process(command, console: false)
|
285
|
+
puts "DEBUG: start_process called with command: #{command}" if OPTIONS[:debug]
|
286
|
+
|
210
287
|
return start_console_process(command) if console
|
211
|
-
|
288
|
+
|
289
|
+
begin
|
290
|
+
stdin, stdout, stderr, wait_thr = Open3.popen3(command)
|
291
|
+
puts "DEBUG: Process started with PID: #{wait_thr.pid}" if OPTIONS[:debug]
|
292
|
+
rescue => e
|
293
|
+
puts "DEBUG: Error starting process: #{e.message}" if OPTIONS[:debug]
|
294
|
+
return nil
|
295
|
+
end
|
212
296
|
|
213
297
|
# Create a monitor thread that will detect if process dies
|
214
298
|
monitor_thread =
|
215
299
|
Thread.new do
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
300
|
+
begin
|
301
|
+
wait_thr.value # Wait for process to finish
|
302
|
+
is_ember = command.include?("ember-cli")
|
303
|
+
@ember_running = false if is_ember
|
304
|
+
@unicorn_running = false unless is_ember
|
305
|
+
GLib::Idle.add do
|
306
|
+
update_tab_labels if @notebook
|
307
|
+
false
|
308
|
+
end
|
309
|
+
rescue => e
|
310
|
+
puts "DEBUG: Error in monitor thread: #{e.message}" if OPTIONS[:debug]
|
223
311
|
end
|
224
312
|
end
|
225
313
|
|
226
|
-
#
|
314
|
+
# Clear the buffer before starting
|
315
|
+
buffer = command.include?("ember-cli") ? @ember_output : @unicorn_output
|
316
|
+
buffer.clear
|
317
|
+
|
318
|
+
# Add a start message to the buffer
|
319
|
+
timestamp = Time.now.strftime("%H:%M:%S")
|
320
|
+
buffer << "#{timestamp} - Starting #{command}...\n"
|
321
|
+
|
322
|
+
# Force immediate GUI update
|
323
|
+
GLib::Idle.add do
|
324
|
+
update_all_views
|
325
|
+
false
|
326
|
+
end
|
327
|
+
|
328
|
+
# Monitor stdout
|
227
329
|
Thread.new do
|
228
|
-
|
229
|
-
|
230
|
-
command.include?("ember-cli") ? @ember_output : @unicorn_output
|
231
|
-
|
232
|
-
|
233
|
-
|
234
|
-
|
330
|
+
begin
|
331
|
+
while line = stdout.gets
|
332
|
+
buffer = command.include?("ember-cli") ? @ember_output : @unicorn_output
|
333
|
+
puts "[OUT] #{line}" if OPTIONS[:debug]
|
334
|
+
|
335
|
+
# Add to buffer with size management
|
336
|
+
buffer << line
|
337
|
+
|
338
|
+
# Print buffer size for debugging
|
339
|
+
if OPTIONS[:debug]
|
340
|
+
puts "DEBUG: Added to buffer: #{line.inspect}"
|
341
|
+
if buffer.size % 10 == 0
|
342
|
+
puts "DEBUG: Buffer size now: #{buffer.size}"
|
343
|
+
end
|
344
|
+
end
|
345
|
+
|
346
|
+
# Trim if needed
|
347
|
+
if buffer.size > BUFFER_SIZE
|
348
|
+
buffer.shift(buffer.size - BUFFER_SIZE)
|
349
|
+
end
|
350
|
+
|
351
|
+
# Force GUI update on main thread
|
352
|
+
GLib::Idle.add do
|
353
|
+
update_all_views
|
354
|
+
false
|
355
|
+
end
|
356
|
+
|
357
|
+
# Also publish to pipe for --attach mode in background
|
358
|
+
publish_to_pipe(line, process: command.include?("ember-cli") ? :ember : :unicorn, stream: :stdout)
|
359
|
+
end
|
360
|
+
rescue => e
|
361
|
+
puts "DEBUG: Error in stdout thread: #{e.message}" if OPTIONS[:debug]
|
362
|
+
puts e.backtrace.join("\n") if OPTIONS[:debug]
|
363
|
+
|
364
|
+
# Add error to buffer for visibility
|
365
|
+
buffer = command.include?("ember-cli") ? @ember_output : @unicorn_output
|
366
|
+
error_msg = "ERROR: Exception in stdout thread: #{e.message}\n"
|
367
|
+
buffer << error_msg
|
368
|
+
|
235
369
|
# Force GUI update
|
236
370
|
GLib::Idle.add do
|
237
371
|
update_all_views
|
@@ -240,15 +374,50 @@ module ::DiscourseSystray
|
|
240
374
|
end
|
241
375
|
end
|
242
376
|
|
243
|
-
# Monitor stderr
|
377
|
+
# Monitor stderr
|
244
378
|
Thread.new do
|
245
|
-
|
246
|
-
|
247
|
-
command.include?("ember-cli") ? @ember_output : @unicorn_output
|
248
|
-
|
249
|
-
|
250
|
-
|
251
|
-
|
379
|
+
begin
|
380
|
+
while line = stderr.gets
|
381
|
+
buffer = command.include?("ember-cli") ? @ember_output : @unicorn_output
|
382
|
+
puts "[ERR] #{line}" if OPTIONS[:debug]
|
383
|
+
|
384
|
+
# Format error line
|
385
|
+
error_line = "E: #{line}"
|
386
|
+
|
387
|
+
# Add to buffer with size management
|
388
|
+
buffer << error_line
|
389
|
+
|
390
|
+
# Print buffer size for debugging
|
391
|
+
if OPTIONS[:debug]
|
392
|
+
puts "DEBUG: Added to buffer: #{error_line.inspect}"
|
393
|
+
if buffer.size % 10 == 0
|
394
|
+
puts "DEBUG: Buffer size now: #{buffer.size}"
|
395
|
+
end
|
396
|
+
end
|
397
|
+
|
398
|
+
# Trim if needed
|
399
|
+
if buffer.size > BUFFER_SIZE
|
400
|
+
buffer.shift(buffer.size - BUFFER_SIZE)
|
401
|
+
end
|
402
|
+
|
403
|
+
# Force GUI update on main thread
|
404
|
+
GLib::Idle.add do
|
405
|
+
update_all_views
|
406
|
+
false
|
407
|
+
end
|
408
|
+
|
409
|
+
# Also publish to pipe for --attach mode in background
|
410
|
+
publish_to_pipe(error_line, process: command.include?("ember-cli") ? :ember : :unicorn, stream: :stderr)
|
411
|
+
end
|
412
|
+
rescue => e
|
413
|
+
puts "DEBUG: Error in stderr thread: #{e.message}" if OPTIONS[:debug]
|
414
|
+
puts e.backtrace.join("\n") if OPTIONS[:debug]
|
415
|
+
|
416
|
+
# Add error to buffer for visibility
|
417
|
+
buffer = command.include?("ember-cli") ? @ember_output : @unicorn_output
|
418
|
+
error_msg = "ERROR: Exception in stderr thread: #{e.message}\n"
|
419
|
+
buffer << error_msg
|
420
|
+
|
252
421
|
# Force GUI update
|
253
422
|
GLib::Idle.add do
|
254
423
|
update_all_views
|
@@ -268,28 +437,43 @@ module ::DiscourseSystray
|
|
268
437
|
end
|
269
438
|
|
270
439
|
def show_status_window
|
440
|
+
puts "DEBUG: show_status_window called" if OPTIONS[:debug]
|
441
|
+
|
271
442
|
if @status_window&.visible?
|
443
|
+
puts "DEBUG: Status window already visible, presenting it" if OPTIONS[:debug]
|
272
444
|
@status_window.present
|
273
445
|
# Force window to current workspace in i3
|
274
446
|
if @status_window.window
|
275
447
|
@status_window.window.raise
|
276
448
|
if system("which i3-msg >/dev/null 2>&1")
|
277
|
-
|
278
|
-
system(
|
279
|
-
"i3-msg '[id=#{@status_window.window.xid}] move workspace current'"
|
280
|
-
)
|
449
|
+
system("i3-msg '[id=#{@status_window.window.xid}] move workspace current'")
|
281
450
|
system("i3-msg '[id=#{@status_window.window.xid}] focus'")
|
282
451
|
end
|
283
452
|
end
|
453
|
+
|
454
|
+
# Force an update of the views even if window is already visible
|
455
|
+
GLib::Idle.add do
|
456
|
+
update_all_views
|
457
|
+
false
|
458
|
+
end
|
459
|
+
|
284
460
|
return
|
285
461
|
end
|
286
462
|
|
287
|
-
# Clean up any existing window
|
463
|
+
# Clean up any existing window and notebook
|
288
464
|
if @status_window
|
465
|
+
puts "DEBUG: Destroying existing status window" if OPTIONS[:debug]
|
466
|
+
@notebook = nil # Clear notebook reference
|
467
|
+
@ember_view = nil
|
468
|
+
@unicorn_view = nil
|
469
|
+
@ember_label = nil
|
470
|
+
@unicorn_label = nil
|
289
471
|
@status_window.destroy
|
290
472
|
@status_window = nil
|
291
473
|
end
|
292
474
|
|
475
|
+
# Create new window and components
|
476
|
+
puts "DEBUG: Creating new status window" if OPTIONS[:debug]
|
293
477
|
@status_window = Gtk::Window.new("Discourse Status")
|
294
478
|
@status_window.set_wmclass("discourse-status", "Discourse Status")
|
295
479
|
|
@@ -299,15 +483,18 @@ module ::DiscourseSystray
|
|
299
483
|
geo = config["window_geometry"]
|
300
484
|
@status_window.move(geo["x"], geo["y"])
|
301
485
|
@status_window.resize(geo["width"], geo["height"])
|
486
|
+
puts "DEBUG: Set window geometry from config: #{geo.inspect}" if OPTIONS[:debug]
|
302
487
|
else
|
303
488
|
@status_window.set_default_size(800, 600)
|
304
489
|
@status_window.window_position = :center
|
490
|
+
puts "DEBUG: Set default window size 800x600" if OPTIONS[:debug]
|
305
491
|
end
|
306
492
|
@status_window.type_hint = :dialog
|
307
493
|
@status_window.set_role("discourse-status-dialog")
|
308
494
|
|
309
495
|
# Handle window destruction and hide
|
310
496
|
@status_window.signal_connect("delete-event") do
|
497
|
+
puts "DEBUG: Window delete-event triggered" if OPTIONS[:debug]
|
311
498
|
save_window_geometry
|
312
499
|
@status_window.hide
|
313
500
|
true # Prevent destruction
|
@@ -319,176 +506,188 @@ module ::DiscourseSystray
|
|
319
506
|
false
|
320
507
|
end
|
321
508
|
|
509
|
+
# Create notebook only if it doesn't exist
|
510
|
+
puts "DEBUG: Creating notebook" if OPTIONS[:debug]
|
322
511
|
@notebook = Gtk::Notebook.new
|
323
512
|
|
324
|
-
|
325
|
-
|
326
|
-
|
513
|
+
# Only create views if they don't exist
|
514
|
+
if @ember_view.nil?
|
515
|
+
puts "DEBUG: Creating ember view" if OPTIONS[:debug]
|
516
|
+
@ember_view = create_log_view(@ember_output)
|
517
|
+
@ember_label = create_status_label("Ember CLI", @ember_running)
|
518
|
+
@notebook.append_page(@ember_view, @ember_label)
|
519
|
+
end
|
327
520
|
|
328
|
-
@unicorn_view
|
329
|
-
|
330
|
-
|
521
|
+
if @unicorn_view.nil?
|
522
|
+
puts "DEBUG: Creating unicorn view" if OPTIONS[:debug]
|
523
|
+
@unicorn_view = create_log_view(@unicorn_output)
|
524
|
+
@unicorn_label = create_status_label("Unicorn", @unicorn_running)
|
525
|
+
@notebook.append_page(@unicorn_view, @unicorn_label)
|
526
|
+
end
|
331
527
|
|
332
528
|
@status_window.add(@notebook)
|
529
|
+
puts "DEBUG: Added notebook to status window" if OPTIONS[:debug]
|
530
|
+
|
333
531
|
@status_window.show_all
|
532
|
+
puts "DEBUG: Called show_all on status window" if OPTIONS[:debug]
|
533
|
+
|
534
|
+
# Force an immediate update of the views
|
535
|
+
GLib::Idle.add do
|
536
|
+
puts "DEBUG: Forcing immediate update after window creation" if OPTIONS[:debug]
|
537
|
+
update_all_views
|
538
|
+
false
|
539
|
+
end
|
334
540
|
end
|
335
541
|
|
336
542
|
def update_all_views
|
543
|
+
puts "DEBUG: update_all_views called" if OPTIONS[:debug]
|
544
|
+
|
545
|
+
# Basic validity checks
|
337
546
|
return unless @status_window && !@status_window.destroyed?
|
338
547
|
return unless @ember_view && @unicorn_view
|
339
|
-
|
340
|
-
return if @ember_view.destroyed? || @unicorn_view.destroyed?
|
341
|
-
return if @ember_view.child.destroyed? || @unicorn_view.child.destroyed?
|
342
|
-
|
548
|
+
|
343
549
|
begin
|
344
|
-
|
550
|
+
# Always update both views for now to ensure content is displayed
|
551
|
+
if @ember_view && !@ember_view.destroyed? && @ember_view.child && !@ember_view.child.destroyed?
|
345
552
|
update_log_view(@ember_view.child, @ember_output)
|
346
553
|
end
|
347
|
-
|
554
|
+
|
555
|
+
if @unicorn_view && !@unicorn_view.destroyed? && @unicorn_view.child && !@unicorn_view.child.destroyed?
|
348
556
|
update_log_view(@unicorn_view.child, @unicorn_output)
|
349
557
|
end
|
350
|
-
|
351
|
-
|
558
|
+
|
559
|
+
# Process any pending GTK events
|
560
|
+
while Gtk.events_pending?
|
561
|
+
Gtk.main_iteration_do(false)
|
562
|
+
end
|
563
|
+
rescue => e
|
564
|
+
puts "DEBUG: Error in update_all_views: #{e.message}" if OPTIONS[:debug]
|
565
|
+
puts e.backtrace.join("\n") if OPTIONS[:debug]
|
352
566
|
end
|
353
567
|
end
|
354
568
|
|
355
569
|
def create_log_view(buffer)
|
570
|
+
puts "DEBUG: create_log_view called for #{buffer == @ember_output ? 'ember' : 'unicorn'}" if OPTIONS[:debug]
|
571
|
+
|
572
|
+
# Create a scrolled window to contain the text view
|
356
573
|
scroll = Gtk::ScrolledWindow.new
|
574
|
+
|
575
|
+
# Create a simple text view with minimal configuration
|
357
576
|
text_view = Gtk::TextView.new
|
358
577
|
text_view.editable = false
|
578
|
+
text_view.cursor_visible = false
|
359
579
|
text_view.wrap_mode = :word
|
360
|
-
|
361
|
-
#
|
580
|
+
|
581
|
+
# Use a fixed-width font
|
582
|
+
text_view.monospace = true
|
583
|
+
|
584
|
+
# Set colors - white text on black background
|
362
585
|
text_view.override_background_color(:normal, Gdk::RGBA.new(0, 0, 0, 1))
|
363
586
|
text_view.override_color(:normal, Gdk::RGBA.new(1, 1, 1, 1))
|
364
|
-
|
365
|
-
#
|
366
|
-
|
367
|
-
|
368
|
-
|
369
|
-
|
370
|
-
|
371
|
-
|
372
|
-
|
587
|
+
|
588
|
+
# Set font size explicitly
|
589
|
+
font_desc = Pango::FontDescription.new
|
590
|
+
font_desc.family = "Monospace"
|
591
|
+
font_desc.size = 12 * Pango::SCALE
|
592
|
+
text_view.override_font(font_desc)
|
593
|
+
|
594
|
+
# Set initial text
|
595
|
+
text_view.buffer.text = "Loading log data...\n"
|
596
|
+
|
597
|
+
# Add the text view to the scrolled window
|
598
|
+
scroll.add(text_view)
|
599
|
+
|
600
|
+
# Set up a timer to update the view more frequently
|
373
601
|
@view_timeouts ||= {}
|
374
|
-
|
375
|
-
|
376
|
-
|
377
|
-
|
378
|
-
|
379
|
-
|
380
|
-
|
381
|
-
|
382
|
-
|
383
|
-
update_log_view(text_view, buffer)
|
384
|
-
rescue StandardError => e
|
385
|
-
puts "Error updating log view: #{e}" if OPTIONS[:debug]
|
386
|
-
end
|
387
|
-
true # Keep the timeout active
|
602
|
+
timeout_id = GLib::Timeout.add(250) do
|
603
|
+
if text_view.destroyed? || scroll.destroyed?
|
604
|
+
@view_timeouts.delete(text_view.object_id)
|
605
|
+
false # Stop the timer
|
606
|
+
else
|
607
|
+
begin
|
608
|
+
update_log_view(text_view, buffer)
|
609
|
+
rescue => e
|
610
|
+
puts "DEBUG: Error updating log view: #{e.message}" if OPTIONS[:debug]
|
388
611
|
end
|
612
|
+
true # Continue the timer
|
389
613
|
end
|
390
|
-
|
614
|
+
end
|
615
|
+
|
616
|
+
# Store the timeout ID for cleanup
|
391
617
|
@view_timeouts[text_view.object_id] = timeout_id
|
392
|
-
|
393
|
-
# Clean up
|
618
|
+
|
619
|
+
# Clean up when the view is destroyed
|
394
620
|
text_view.signal_connect("destroy") do
|
395
|
-
if
|
396
|
-
|
397
|
-
GLib::Source.remove(timeout_id)
|
398
|
-
rescue StandardError
|
399
|
-
nil
|
400
|
-
end
|
621
|
+
if id = @view_timeouts.delete(text_view.object_id)
|
622
|
+
GLib::Source.remove(id) rescue nil
|
401
623
|
end
|
402
624
|
end
|
403
|
-
|
404
|
-
|
625
|
+
|
626
|
+
# Do an initial update
|
627
|
+
update_log_view(text_view, buffer)
|
628
|
+
|
629
|
+
# Return the scrolled window
|
405
630
|
scroll
|
406
631
|
end
|
407
632
|
|
408
|
-
|
409
|
-
# Basic ANSI colors
|
410
|
-
{
|
411
|
-
"31" => "#ff6b6b", # Brighter red
|
412
|
-
"32" => "#87ff87", # Brighter green
|
413
|
-
"33" => "#ffff87", # Brighter yellow
|
414
|
-
"34" => "#87d7ff", # Brighter blue
|
415
|
-
"35" => "#ff87ff", # Brighter magenta
|
416
|
-
"36" => "#87ffff", # Brighter cyan
|
417
|
-
"37" => "#ffffff" # White
|
418
|
-
}.each do |code, color|
|
419
|
-
buffer.create_tag("ansi_#{code}", foreground: color)
|
420
|
-
end
|
421
|
-
|
422
|
-
# Add more tags for bold, etc
|
423
|
-
buffer.create_tag("bold", weight: :bold)
|
424
|
-
end
|
633
|
+
# We're not using ANSI tags anymore since we're stripping ANSI codes
|
425
634
|
|
426
635
|
def update_log_view(text_view, buffer)
|
636
|
+
puts "DEBUG: update_log_view called" if OPTIONS[:debug]
|
637
|
+
|
638
|
+
# Basic validity checks
|
427
639
|
return if text_view.nil? || text_view.destroyed?
|
428
640
|
return if text_view.buffer.nil? || text_view.buffer.destroyed?
|
429
641
|
|
430
|
-
#
|
431
|
-
|
432
|
-
|
433
|
-
|
434
|
-
|
435
|
-
|
436
|
-
:@unicorn_line_count
|
437
|
-
end
|
438
|
-
)
|
439
|
-
current_offset = instance_variable_get(offset_var)
|
440
|
-
|
441
|
-
# Don't call if we've already processed all lines
|
442
|
-
return if buffer.size <= current_offset
|
443
|
-
|
444
|
-
adj = text_view&.parent&.vadjustment
|
445
|
-
was_at_bottom = (adj && adj.value >= adj.upper - adj.page_size - 50)
|
446
|
-
old_value = adj ? adj.value : 0
|
447
|
-
|
448
|
-
# Process only the new lines
|
449
|
-
new_lines = buffer[current_offset..-1]
|
450
|
-
new_lines.each do |line|
|
451
|
-
ansi_segments =
|
452
|
-
line.scan(/\e\[([0-9;]*)m([^\e]*)|\e\[K([^\e]*)|([^\e]+)/)
|
453
|
-
|
454
|
-
ansi_segments.each do |codes, text_part, clear_part, plain|
|
455
|
-
chunk = text_part || clear_part || plain.to_s
|
456
|
-
chunk_start_iter = text_view.buffer.end_iter
|
457
|
-
text_view.buffer.insert(chunk_start_iter, chunk)
|
458
|
-
|
459
|
-
# For each ANSI code, apply tags
|
460
|
-
if codes
|
461
|
-
codes
|
462
|
-
.split(";")
|
463
|
-
.each do |code|
|
464
|
-
case code
|
465
|
-
when "1"
|
466
|
-
text_view.buffer.apply_tag(
|
467
|
-
"bold",
|
468
|
-
chunk_start_iter,
|
469
|
-
text_view.buffer.end_iter
|
470
|
-
)
|
471
|
-
when "31".."37"
|
472
|
-
text_view.buffer.apply_tag(
|
473
|
-
"ansi_#{code}",
|
474
|
-
chunk_start_iter,
|
475
|
-
text_view.buffer.end_iter
|
476
|
-
)
|
477
|
-
end
|
478
|
-
end
|
479
|
-
end
|
642
|
+
# Debug buffer content
|
643
|
+
if OPTIONS[:debug]
|
644
|
+
puts "DEBUG: Buffer size: #{buffer.size}"
|
645
|
+
if buffer.size > 0
|
646
|
+
puts "DEBUG: First line: #{buffer.first.inspect}"
|
647
|
+
puts "DEBUG: Last line: #{buffer.last.inspect}"
|
480
648
|
end
|
481
649
|
end
|
482
650
|
|
483
|
-
#
|
484
|
-
|
651
|
+
# If buffer is empty, add a placeholder message
|
652
|
+
if buffer.empty?
|
653
|
+
buffer << "No log data available yet. Start Discourse to see logs.\n"
|
654
|
+
end
|
485
655
|
|
486
|
-
#
|
487
|
-
|
488
|
-
|
656
|
+
# Completely replace the buffer content with all lines
|
657
|
+
begin
|
658
|
+
# Make a local copy of the buffer to avoid race conditions
|
659
|
+
buffer_copy = buffer.dup
|
660
|
+
|
661
|
+
# Join all buffer lines into a single string
|
662
|
+
all_content = buffer_copy.join("")
|
663
|
+
|
664
|
+
# Strip ANSI codes
|
665
|
+
clean_content = all_content.gsub(/\e\[[0-9;]*[mK]/, '')
|
666
|
+
|
667
|
+
# Always update the content to ensure it's displayed
|
668
|
+
text_view.buffer.text = clean_content
|
669
|
+
|
670
|
+
puts "DEBUG: Updated buffer text (#{clean_content.length} chars)" if OPTIONS[:debug]
|
671
|
+
|
672
|
+
# Scroll to bottom
|
673
|
+
adj = text_view&.parent&.vadjustment
|
674
|
+
if adj
|
489
675
|
adj.value = adj.upper - adj.page_size
|
490
|
-
|
491
|
-
|
676
|
+
end
|
677
|
+
|
678
|
+
# Process any pending GTK events to ensure UI updates
|
679
|
+
while Gtk.events_pending?
|
680
|
+
Gtk.main_iteration_do(false)
|
681
|
+
end
|
682
|
+
rescue => e
|
683
|
+
puts "DEBUG: Error updating text view: #{e.message}" if OPTIONS[:debug]
|
684
|
+
puts e.backtrace.join("\n") if OPTIONS[:debug]
|
685
|
+
|
686
|
+
# Try a fallback approach
|
687
|
+
begin
|
688
|
+
text_view.buffer.text = "Error displaying log. See console for details.\n#{e.message}"
|
689
|
+
rescue => e2
|
690
|
+
puts "DEBUG: Even fallback approach failed: #{e2.message}" if OPTIONS[:debug]
|
492
691
|
end
|
493
692
|
end
|
494
693
|
end
|
@@ -624,6 +823,12 @@ module ::DiscourseSystray
|
|
624
823
|
if OPTIONS[:attach]
|
625
824
|
require "rb-inotify"
|
626
825
|
|
826
|
+
# Set up signal handling for Ctrl+C
|
827
|
+
Signal.trap("INT") do
|
828
|
+
puts "Received interrupt signal, shutting down..."
|
829
|
+
exit 0
|
830
|
+
end
|
831
|
+
|
627
832
|
notifier = INotify::Notifier.new
|
628
833
|
|
629
834
|
begin
|
@@ -638,34 +843,57 @@ module ::DiscourseSystray
|
|
638
843
|
end
|
639
844
|
|
640
845
|
# Read from pipe in a separate thread
|
641
|
-
reader =
|
642
|
-
|
643
|
-
|
644
|
-
|
645
|
-
|
646
|
-
|
846
|
+
reader = Thread.new do
|
847
|
+
# Set thread abort on exception
|
848
|
+
Thread.current.abort_on_exception = true
|
849
|
+
|
850
|
+
begin
|
851
|
+
while true
|
852
|
+
begin
|
853
|
+
# Use non-blocking read with timeout
|
854
|
+
ready = IO.select([pipe], nil, nil, 0.1)
|
855
|
+
if ready && ready[0].include?(pipe)
|
856
|
+
line = pipe.gets
|
857
|
+
if line
|
647
858
|
puts line
|
648
859
|
STDOUT.flush
|
649
860
|
end
|
650
861
|
end
|
862
|
+
rescue IOError, Errno::EBADF => e
|
863
|
+
puts "DEBUG: Pipe read error: #{e.message}" if OPTIONS[:debug]
|
864
|
+
break
|
865
|
+
end
|
651
866
|
|
652
|
-
|
653
|
-
|
654
|
-
|
655
|
-
|
656
|
-
end
|
867
|
+
# Check if pipe still exists
|
868
|
+
unless File.exist?(PIPE_PATH)
|
869
|
+
puts "Pipe was deleted, exiting."
|
870
|
+
exit 0
|
657
871
|
end
|
658
|
-
|
659
|
-
|
660
|
-
|
872
|
+
|
873
|
+
# Small sleep to prevent CPU hogging
|
874
|
+
sleep 0.01
|
661
875
|
end
|
876
|
+
rescue EOFError, IOError
|
877
|
+
puts "Pipe closed, exiting."
|
878
|
+
exit 0
|
662
879
|
end
|
663
|
-
|
664
|
-
|
665
|
-
notifier
|
880
|
+
end
|
881
|
+
|
882
|
+
# Set up non-blocking notifier processing
|
883
|
+
# Instead of notifier.run which blocks indefinitely, use a loop with timeout
|
884
|
+
while true
|
885
|
+
# Process any pending inotify events, with timeout
|
886
|
+
notifier.process
|
887
|
+
|
888
|
+
# Sleep briefly to prevent CPU hogging
|
889
|
+
sleep 0.1
|
890
|
+
end
|
666
891
|
rescue Errno::ENOENT
|
667
892
|
puts "Pipe doesn't exist, exiting."
|
668
893
|
exit 1
|
894
|
+
rescue Interrupt
|
895
|
+
puts "Interrupted, exiting."
|
896
|
+
exit 0
|
669
897
|
ensure
|
670
898
|
reader&.kill
|
671
899
|
pipe&.close
|
@@ -695,21 +923,52 @@ module ::DiscourseSystray
|
|
695
923
|
|
696
924
|
# Setup systray icon and menu
|
697
925
|
init_systray
|
698
|
-
|
926
|
+
|
699
927
|
# Start GTK main loop
|
700
928
|
Gtk.main
|
701
929
|
end
|
702
930
|
end
|
703
931
|
|
704
|
-
|
705
|
-
|
706
|
-
|
707
|
-
|
708
|
-
|
709
|
-
|
710
|
-
|
932
|
+
# Queue for pipe messages to avoid blocking
|
933
|
+
def initialize_pipe_queue
|
934
|
+
@pipe_queue = Queue.new
|
935
|
+
@pipe_thread = Thread.new do
|
936
|
+
loop do
|
937
|
+
begin
|
938
|
+
msg = @pipe_queue.pop
|
939
|
+
break if msg == :exit
|
940
|
+
|
941
|
+
if File.exist?(PIPE_PATH)
|
942
|
+
begin
|
943
|
+
# Use non-blocking write with timeout
|
944
|
+
Timeout.timeout(0.5) do
|
945
|
+
File.open(PIPE_PATH, "w") do |f|
|
946
|
+
f.puts(msg)
|
947
|
+
f.flush
|
948
|
+
end
|
949
|
+
end
|
950
|
+
rescue Timeout::Error
|
951
|
+
puts "Timeout writing to pipe" if OPTIONS[:debug]
|
952
|
+
rescue Errno::EPIPE, IOError => e
|
953
|
+
puts "Error writing to pipe: #{e}" if OPTIONS[:debug]
|
954
|
+
end
|
955
|
+
end
|
956
|
+
rescue => e
|
957
|
+
puts "Error in pipe thread: #{e}" if OPTIONS[:debug]
|
958
|
+
end
|
959
|
+
|
960
|
+
# Small sleep to prevent CPU hogging
|
961
|
+
sleep 0.01
|
962
|
+
end
|
711
963
|
end
|
712
964
|
end
|
965
|
+
|
966
|
+
def publish_to_pipe(msg, process: nil, stream: nil)
|
967
|
+
source_info = "[#{process || 'unknown'}:#{stream || 'unknown'}]"
|
968
|
+
puts "Publish to pipe #{source_info}: #{msg}" if OPTIONS[:debug]
|
969
|
+
|
970
|
+
@pipe_queue.push(msg) if @pipe_queue
|
971
|
+
end
|
713
972
|
|
714
973
|
def handle_command(cmd)
|
715
974
|
puts "Received command: #{cmd}" if OPTIONS[:debug]
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: discourse-systray
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.1.
|
4
|
+
version: 0.1.4
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Sam Saffron
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2025-
|
11
|
+
date: 2025-03-08 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: gtk3
|