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.
Files changed (3) hide show
  1. checksums.yaml +4 -4
  2. data/lib/discourse_systray/systray.rb +440 -181
  3. metadata +2 -2
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 5210b401c239734ab826f42710f3d925e65e52d62b9a5f8b3da18d56b874469b
4
- data.tar.gz: 838fac902e2cad6158ea5c5f52086047f4687b5871e1ce5c4b4989203d6e3eda
3
+ metadata.gz: 52f16a25286153255ec00c850f08e74cd2b704903d8adf850705f17aa6898fdb
4
+ data.tar.gz: 4f708705f6669dff35a737ff328917c3b603b7cc7f87c0c1d3a7602ee0157852
5
5
  SHA512:
6
- metadata.gz: a4694037048949cbb48696ff280327b1d1f6f6349685de9e8a55fcfc81f27326e4812e526c0976c7615a0d0076543139c158744cd451841503b32f70d0963597
7
- data.tar.gz: 7193b7995de0c0ff81ca6224c12d478b916bb3972ec896a4266cfbf7a12dc1881091f9cc28c434d3fc65f190e0902e031dbcca6bb66ff705cd92f7891a53e766
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 = 2000
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
- stdin, stdout, stderr, wait_thr = Open3.popen3(command)
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
- wait_thr.value # Wait for process to finish
217
- is_ember = command.include?("ember-cli")
218
- @ember_running = false if is_ember
219
- @unicorn_running = false unless is_ember
220
- GLib::Idle.add do
221
- update_tab_labels if @notebook
222
- false
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
- # Monitor stdout - send to both console and UX buffer
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
- while line = stdout.gets
229
- buffer =
230
- command.include?("ember-cli") ? @ember_output : @unicorn_output
231
- publish_to_pipe(line)
232
- puts "[OUT] #{line}" if OPTIONS[:debug]
233
- buffer << line
234
- buffer.shift if buffer.size > BUFFER_SIZE
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 - send to both console and UX buffer
377
+ # Monitor stderr
244
378
  Thread.new do
245
- while line = stderr.gets
246
- buffer =
247
- command.include?("ember-cli") ? @ember_output : @unicorn_output
248
- publish_to_pipe("ERROR: #{line}")
249
- puts "[ERR] #{line}" if OPTIONS[:debug]
250
- buffer << line
251
- buffer.shift if buffer.size > BUFFER_SIZE
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
- # First move to current workspace, then focus
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
- @ember_view = create_log_view(@ember_output)
325
- @ember_label = create_status_label("Ember CLI", @ember_running)
326
- @notebook.append_page(@ember_view, @ember_label)
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 = create_log_view(@unicorn_output)
329
- @unicorn_label = create_status_label("Unicorn", @unicorn_running)
330
- @notebook.append_page(@unicorn_view, @unicorn_label)
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
- return unless @ember_view.child && @unicorn_view.child
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
- if @ember_view.visible? && @ember_view.child.visible?
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
- if @unicorn_view.visible? && @unicorn_view.child.visible?
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
- rescue StandardError => e
351
- puts "Error updating views: #{e}" if OPTIONS[:debug]
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
- # Set white text on black background
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
- # Create text tags for colors
366
- _tag_table = text_view.buffer.tag_table
367
- create_ansi_tags(text_view.buffer)
368
-
369
- # Initial text
370
- update_log_view(text_view, buffer)
371
-
372
- # Store timeouts in instance variable for proper cleanup
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
- # Set up periodic refresh with validity check
376
- timeout_id =
377
- GLib::Timeout.add(500) do
378
- if text_view&.parent.nil? || text_view.destroyed?
379
- @view_timeouts.delete(text_view.object_id)
380
- false # Stop the timeout if view is destroyed
381
- else
382
- begin
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 timeout when view is destroyed
618
+
619
+ # Clean up when the view is destroyed
394
620
  text_view.signal_connect("destroy") do
395
- if timeout_id = @view_timeouts.delete(text_view.object_id)
396
- begin
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
- scroll.add(text_view)
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
- def create_ansi_tags(buffer)
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
- # Determine which offset counter to use
431
- offset_var =
432
- (
433
- if buffer.equal?(@ember_output)
434
- :@ember_line_count
435
- else
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
- # Update our offset counter
484
- instance_variable_set(offset_var, buffer.size)
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
- # Restore scroll position
487
- if adj
488
- if was_at_bottom
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
- else
491
- adj.value = old_value
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
- Thread.new do
643
- begin
644
- while true
645
- if IO.select([pipe], nil, nil, 0.5)
646
- while line = pipe.gets
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
- sleep 0.1
653
- unless File.exist?(PIPE_PATH)
654
- puts "Pipe was deleted, exiting."
655
- exit 0
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
- rescue EOFError, IOError
659
- puts "Pipe closed, exiting."
660
- exit 0
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
- # Handle notifications in main thread
665
- notifier.run
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
- def publish_to_pipe(msg)
705
- return unless File.exist?(PIPE_PATH)
706
- puts "Publish to pipe: #{msg}" if OPTIONS[:debug]
707
- begin
708
- File.open(PIPE_PATH, "w") { |f| f.puts(msg) }
709
- rescue Errno::EPIPE, IOError => e
710
- puts "Error writing to pipe: #{e}" if OPTIONS[:debug]
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.2
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-01-26 00:00:00.000000000 Z
11
+ date: 2025-03-08 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: gtk3