onlylogs 0.1.2

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 (54) hide show
  1. checksums.yaml +7 -0
  2. data/README.md +311 -0
  3. data/Rakefile +8 -0
  4. data/app/assets/config/onlylogs_manifest.js +2 -0
  5. data/app/assets/images/onlylogs/favicon/apple-touch-icon.png +0 -0
  6. data/app/assets/images/onlylogs/favicon/favicon-96x96.png +0 -0
  7. data/app/assets/images/onlylogs/favicon/favicon.ico +0 -0
  8. data/app/assets/images/onlylogs/favicon/favicon.svg +3 -0
  9. data/app/assets/images/onlylogs/favicon/site.webmanifest.erb +21 -0
  10. data/app/assets/images/onlylogs/favicon/web-app-manifest-192x192.png +0 -0
  11. data/app/assets/images/onlylogs/favicon/web-app-manifest-512x512.png +0 -0
  12. data/app/assets/images/onlylogs/logo.png +0 -0
  13. data/app/channels/onlylogs/application_cable/channel.rb +11 -0
  14. data/app/channels/onlylogs/logs_channel.rb +181 -0
  15. data/app/controllers/onlylogs/application_controller.rb +22 -0
  16. data/app/controllers/onlylogs/logs_controller.rb +23 -0
  17. data/app/helpers/onlylogs/application_helper.rb +4 -0
  18. data/app/javascript/onlylogs/application.js +1 -0
  19. data/app/javascript/onlylogs/controllers/application.js +9 -0
  20. data/app/javascript/onlylogs/controllers/index.js +11 -0
  21. data/app/javascript/onlylogs/controllers/keyboard_shortcuts_controller.js +46 -0
  22. data/app/javascript/onlylogs/controllers/log_streamer_controller.js +432 -0
  23. data/app/javascript/onlylogs/controllers/text_selection_controller.js +90 -0
  24. data/app/jobs/onlylogs/application_job.rb +4 -0
  25. data/app/models/onlylogs/ansi_color_parser.rb +78 -0
  26. data/app/models/onlylogs/application_record.rb +5 -0
  27. data/app/models/onlylogs/batch_sender.rb +61 -0
  28. data/app/models/onlylogs/file.rb +151 -0
  29. data/app/models/onlylogs/file_path_parser.rb +118 -0
  30. data/app/models/onlylogs/grep.rb +54 -0
  31. data/app/models/onlylogs/log_line.rb +24 -0
  32. data/app/models/onlylogs/secure_file_path.rb +31 -0
  33. data/app/views/home/show.html.erb +10 -0
  34. data/app/views/layouts/onlylogs/application.html.erb +27 -0
  35. data/app/views/onlylogs/logs/index.html.erb +49 -0
  36. data/app/views/onlylogs/shared/_log_container.html.erb +106 -0
  37. data/app/views/onlylogs/shared/_log_container_styles.html.erb +228 -0
  38. data/config/importmap.rb +6 -0
  39. data/config/puma_plugins/vector.rb +94 -0
  40. data/config/routes.rb +4 -0
  41. data/config/udp_logger.rb +40 -0
  42. data/config/vector.toml +32 -0
  43. data/db/migrate/20250902112548_create_books.rb +9 -0
  44. data/lib/onlylogs/configuration.rb +133 -0
  45. data/lib/onlylogs/engine.rb +39 -0
  46. data/lib/onlylogs/formatter.rb +14 -0
  47. data/lib/onlylogs/log_silencer_middleware.rb +26 -0
  48. data/lib/onlylogs/logger.rb +10 -0
  49. data/lib/onlylogs/socket_logger.rb +71 -0
  50. data/lib/onlylogs/version.rb +3 -0
  51. data/lib/onlylogs.rb +17 -0
  52. data/lib/puma/plugin/onlylogs_sidecar.rb +113 -0
  53. data/lib/tasks/onlylogs_tasks.rake +4 -0
  54. metadata +110 -0
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: 0f2f05cff2b5f48634427295d3704774b1431c85f5bbba89a5adc9d3a6769c79
4
+ data.tar.gz: 16fc0ca2ec651f9523e266ec883a796f2dc94bb46b952b5b6aaf0eb527fd8871
5
+ SHA512:
6
+ metadata.gz: 7625fcd6b6b275a4008e88d7ec9a085b8277854f759472a7eea0acb98d2b8d81c14023ddb8eae313f74bf95d82b9ce8d134fdc0b52af019d84d560a2e62fb7e8
7
+ data.tar.gz: 104a06ec0c73c0885a89cce09dd25597d4c013c2cac9ab9b88f32a77d785aba323885f6eb6cc7a28bfe87c0e0792777d1aae16e6188ffb7bcc589961b8e7a4f6
data/README.md ADDED
@@ -0,0 +1,311 @@
1
+ <img alt="w:100px" src="app/assets/images/onlylogs/logo.png" width="400px"/>
2
+
3
+ We believe logs are enough.
4
+
5
+ We believe logs in human-readable format are enough.
6
+
7
+ Stop streaming your logs to very expensive external services: just store your logs on disk.
8
+
9
+ When your application grows and you don't want to self-host your log files anymore, you can
10
+ stream them to https://onlylogs.io and continue enjoying the same features.
11
+
12
+ > [!IMPORTANT]
13
+ > https://onlylogs.io is still in beta. Send us an email to a@renuo.ch if you want access to the platform.
14
+
15
+ ## Installation
16
+
17
+ Add this line to your application's Gemfile:
18
+
19
+ ```ruby
20
+ gem "onlylogs"
21
+ ```
22
+
23
+ And then execute:
24
+
25
+ ```bash
26
+ $ bundle
27
+ ```
28
+
29
+ mount the engine in your `routes.rb`
30
+
31
+ ```ruby
32
+ Rails.application.routes.draw do
33
+ # ...
34
+ mount Onlylogs::Engine, at: "/onlylogs"
35
+ ```
36
+
37
+ Finally, you **must secure the engine**. Read the section dedicated to the [Authentication](#authentication).
38
+
39
+ > [!TIP]
40
+ > **Install ripgrep for Better Performance**.
41
+ > For optimal search performance, we recommend installing [ripgrep](https://github.com/BurntSushi/ripgrep).
42
+ > Onlylogs will automatically detect and use ripgrep if available.
43
+
44
+ ## Usage
45
+
46
+ Head to `/onlylogs` and enjoy your logs streamed right into your face!
47
+
48
+ Here you can grep your logs with regular expressions.
49
+
50
+ > [!TIP]
51
+ > Onlylogs automatically detects and uses [ripgrep (rg)](https://github.com/BurntSushi/ripgrep) if available, which provides significantly faster search experience.
52
+ > If ripgrep is not installed, onlylogs falls back to `grep`.
53
+ > A warning icon (⚠️) will be displayed in the toolbar when using `grep` to indicate slower search performance.
54
+
55
+ ## Authentication
56
+
57
+ Yes, we should do this right away, because this engine gives access to your log files, so you want to be sure.
58
+
59
+ The engine has one Controller and one ActionCable channel that **must be protected**.
60
+
61
+ Please be sure to secure them properly.
62
+
63
+ > [!IMPORTANT]
64
+ > By default, onlylogs endpoints are completely inaccessible until basic auth credentials are configured.
65
+
66
+ ### Basic Authentication Setup
67
+
68
+ Credentials can be configured using environment variables, Rails credentials, or programmatically.
69
+ Environment variables take precedence over Rails credentials.
70
+
71
+ ```bash
72
+ # env variables
73
+ export ONLYLOGS_BASIC_AUTH_USER="your_username"
74
+ export ONLYLOGS_BASIC_AUTH_PASSWORD="your_password"
75
+ ```
76
+
77
+ ```yml
78
+ # config/credentials.yml.enc
79
+ onlylogs:
80
+ basic_auth_user: your_username
81
+ basic_auth_password: your_password
82
+ ```
83
+
84
+ ```ruby
85
+ # config/initializers/onlylogs.rb
86
+ Onlylogs.configure do |config|
87
+ config.basic_auth_user = "your_username"
88
+ config.basic_auth_password = "your_password"
89
+ end
90
+ ```
91
+
92
+
93
+ ### Custom Authentication
94
+
95
+ When you need custom authentication logic beyond basic auth,
96
+ you can override the default authentication by configuring a parent controller that defines the `authenticate_onlylogs_user!` method.
97
+
98
+ Configure a custom parent controller in your initializer:
99
+
100
+ ```ruby
101
+ # config/initializers/onlylogs.rb
102
+ Onlylogs.configure do |config|
103
+ config.disable_basic_authentication = true
104
+ config.parent_controller = "ApplicationController" # or any other controller
105
+ end
106
+ ```
107
+
108
+ In your parent controller, define the `authenticate_onlylogs_user!` method:
109
+
110
+ ```ruby
111
+ # app/controllers/application_controller.rb
112
+ class ApplicationController < ActionController::Base
113
+ private
114
+
115
+ def authenticate_onlylogs_user!
116
+ raise unless current_user.can_access_logs?
117
+ end
118
+ end
119
+ ```
120
+
121
+ #### Disabling Authentication
122
+
123
+ For development you can disable basic authentication entirely:
124
+
125
+ ```ruby
126
+ # config/environments/development.rb
127
+ Onlylogs.configure do |config|
128
+ config.disable_basic_authentication = true
129
+ end
130
+ ```
131
+
132
+ ### WebSocket Authentication
133
+
134
+ Logs are streamed through a WebSocket connection, the Websocket is not protected, but in order to stream a file,
135
+ the file path must be white-listed (see section below) and the file path encrypted using `Onlylogs::SecureFilePath.encrypt`
136
+
137
+
138
+ ## Customization
139
+
140
+ Onlylogs provides two ways to customize the appearance of the log viewer: CSS Variables and a complete style override.
141
+ Check the file [_log_container_styles.html.erb](app/views/onlylogs/shared/_log_container_styles.html.erb) for the complete list of CSS variables.
142
+
143
+ ## Configuration
144
+
145
+ Check `configuration.rb` to see a list of all possible configuration.
146
+
147
+ ### File Access Security
148
+
149
+ Onlylogs includes a secure file access system that prevents unauthorized access to files on your server.
150
+ By default, onlylogs can access your Rails environment-specific log files (e.g., `log/development.log`, `log/production.log`).
151
+
152
+ #### Configuring Allowed Files
153
+
154
+ You can configure which files onlylogs is allowed to access by creating a configuration initializer:
155
+
156
+ ```ruby
157
+ # config/initializers/onlylogs.rb
158
+ Onlylogs.configure do |config|
159
+ config.allowed_files = [
160
+ # Default Rails log files
161
+ Rails.root.join("log/development.log"),
162
+ Rails.root.join("log/production.log"),
163
+ Rails.root.join("log/test.log"),
164
+
165
+ # Custom log files
166
+ Rails.root.join("log/custom.log"),
167
+ Rails.root.join("log/api.log"),
168
+
169
+ # Application-specific logs
170
+ Rails.root.join("log/background_jobs.log"),
171
+ Rails.root.join("log/imports.log"),
172
+
173
+ # Allow all .log files in a directory using glob patterns
174
+ Rails.root.join("log/*.log"),
175
+ Rails.root.join("tmp/logs/*.log")
176
+ ]
177
+ end
178
+ ```
179
+
180
+ **Default Behavior:**
181
+ - If not configured, onlylogs defaults to `Rails.root.join("log/#{Rails.env}.log").to_s`
182
+ - This means it will use `log/development.log` in development, `log/production.log` in production, etc.
183
+
184
+ #### Glob Pattern Support
185
+
186
+ Onlylogs supports glob patterns to allow multiple files at once:
187
+
188
+ ```ruby
189
+ # config/initializers/onlylogs.rb
190
+ Onlylogs.configure do |config|
191
+ config.allowed_files = [
192
+ # Allow all .log files in the log directory
193
+ Rails.root.join("log/*.log"),
194
+
195
+ # Allow specific pattern matches
196
+ Rails.root.join("log/*production*.log"),
197
+ Rails.root.join("log/*development*.log"),
198
+
199
+ # Allow files in subdirectories
200
+ Rails.root.join("log/**/*.log"),
201
+ Rails.root.join("tmp/**/*.log")
202
+ ]
203
+ end
204
+ ```
205
+
206
+ **Supported Glob Patterns:**
207
+ - `*.log` - Matches all files ending with `.log` in the specified directory
208
+ - `*production*.log` - Matches files containing "production" and ending with `.log`
209
+ - `**/*.log` - Matches all `.log` files in the directory and all subdirectories
210
+
211
+ **Important Notes:**
212
+ - Patterns are directory-specific - `log/*.log` only matches files in the `log/` directory
213
+ - Multiple patterns can be combined in the same configuration
214
+
215
+
216
+ ### Configuring Code Editor for File Path Links
217
+
218
+ Onlylogs automatically detects file paths in log messages and converts them into clickable links that open in your preferred code editor.
219
+
220
+ For a complete list of supported editors, see [lib/onlylogs/editor_detector.rb](lib/onlylogs/editor_detector.rb).
221
+
222
+ ```bash
223
+ # env variables
224
+ export EDITOR="vscode"
225
+ export RAILS_EDITOR="vscode"
226
+ export ONLYLOGS_EDITOR="vscode" # highest precedence
227
+ ```
228
+
229
+ ```yml
230
+ # config/credentials.yml.enc
231
+ onlylogs:
232
+ editor: vscode
233
+ ```
234
+
235
+ ```ruby
236
+ # config/initializers/onlylogs.rb
237
+ Onlylogs.configure do |config|
238
+ config.editor = :vscode
239
+ end
240
+ ```
241
+
242
+ #### Configuring Maximum Search Results
243
+
244
+ By default, onlylogs limits search results to 100,000 lines to prevent memory issues and ensure responsive performance. You can configure this limit based on your needs:
245
+
246
+ ```ruby
247
+ # config/initializers/onlylogs.rb
248
+ Onlylogs.configure do |config|
249
+ # Set a custom limit (e.g., 50,000 lines)
250
+ config.max_line_matches = 50_000
251
+
252
+ # Or remove the limit entirely (use with caution)
253
+ config.max_line_matches = nil
254
+ end
255
+ ```
256
+
257
+ ## Development & Contributing
258
+
259
+ You are more than welcome to help and contribute to this package.
260
+
261
+ The app uses minitest and includes a dummy app, so getting started should be straightforward.
262
+
263
+ ### Latency Simulation
264
+
265
+ For testing how onlylogs behaves under production-like network conditions, you can simulate latency for HTTP requests and WebSocket connections using the included latency simulation tool.
266
+
267
+ ### Usage
268
+
269
+ ```bash
270
+ # Enable latency simulation (120±30ms jitter on port 3000)
271
+ ./bin/simulate_latency enable
272
+
273
+ # Enable custom latency simulation (150±30ms jitter on port 3000)
274
+ ./bin/simulate_latency enable 150
275
+
276
+ # Enable custom latency and jitter (200±50ms jitter on port 3000)
277
+ ./bin/simulate_latency enable 200/50
278
+
279
+ # Enable latency simulation on custom port (120±30ms jitter on port 8080)
280
+ ./bin/simulate_latency enable -p 8080
281
+
282
+ # Enable custom latency and jitter on custom port (150±50ms jitter on port 8080)
283
+ ./bin/simulate_latency enable 150/50 -p 8080
284
+
285
+ # Test the latency
286
+ ./bin/simulate_latency test
287
+
288
+ # Check current status
289
+ ./bin/simulate_latency status
290
+
291
+ # Disable and clean up
292
+ ./bin/simulate_latency disable
293
+ ```
294
+
295
+ ### Plans for the future
296
+
297
+ We believe that by simply analysing your logs you can also have a fancy errors report.
298
+ Yes, correct. You don't need Sentry either.
299
+
300
+ And you know what? You can get also performance reports.
301
+
302
+ All of a sudden you are 100% free from external services for three more things:
303
+
304
+ * logs
305
+ * errors
306
+ * performance
307
+
308
+
309
+ ## License
310
+
311
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
data/Rakefile ADDED
@@ -0,0 +1,8 @@
1
+ require "bundler/setup"
2
+
3
+ APP_RAKEFILE = File.expand_path("test/dummy/Rakefile", __dir__)
4
+ load "rails/tasks/engine.rake"
5
+
6
+ load "rails/tasks/statistics.rake"
7
+
8
+ require "bundler/gem_tasks"
@@ -0,0 +1,2 @@
1
+ //= link_directory ../../javascript/onlylogs .js
2
+ //= link_directory ../../javascript/onlylogs/controllers .js
@@ -0,0 +1,3 @@
1
+ <svg xmlns="http://www.w3.org/2000/svg" version="1.1" xmlns:xlink="http://www.w3.org/1999/xlink" width="286" height="233" viewBox="0 0 286 233"><image width="286" height="233" xlink:href=""></image><style>@media (prefers-color-scheme: light) { :root { filter: none; } }
2
+ @media (prefers-color-scheme: dark) { :root { filter: none; } }
3
+ </style></svg>
@@ -0,0 +1,21 @@
1
+ {
2
+ "name": "Onlylogs",
3
+ "short_name": "Onlylogs",
4
+ "icons": [
5
+ {
6
+ "src": "<%= asset_path 'favicon/web-app-manifest-192x192.png' %>",
7
+ "sizes": "192x192",
8
+ "type": "image/png",
9
+ "purpose": "maskable"
10
+ },
11
+ {
12
+ "src": "<%= asset_path 'favicon/web-app-manifest-512x512.png' %>",
13
+ "sizes": "512x512",
14
+ "type": "image/png",
15
+ "purpose": "maskable"
16
+ }
17
+ ],
18
+ "theme_color": "#ffffff",
19
+ "background_color": "#ffffff",
20
+ "display": "standalone"
21
+ }
Binary file
@@ -0,0 +1,11 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Onlylogs
4
+ module ApplicationCable
5
+ class Channel < ActionCable::Channel::Base
6
+ def subscribed
7
+ stream_from "onlylogs:stream"
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,181 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Onlylogs
4
+ class LogsChannel < ActionCable::Channel::Base
5
+ def subscribed
6
+ # Rails.logger.info "Client subscribed to Onlylogs::LogsChannel"
7
+ # Wait for the client to send the cursor position
8
+ # start_log_watcher will be called from the initialize_watcher method
9
+ end
10
+
11
+ def initialize_watcher(data)
12
+ cleanup_existing_operations
13
+
14
+ # Decrypt and verify the file path
15
+ begin
16
+ encrypted_file_path = data["file_path"]
17
+ if encrypted_file_path.present?
18
+ file_path = Onlylogs::SecureFilePath.decrypt(encrypted_file_path)
19
+
20
+ # Verify the decrypted path is still allowed
21
+ unless Onlylogs.allowed_file_path?(file_path)
22
+ Rails.logger.error "Onlylogs: Attempted to access non-allowed file: #{file_path}"
23
+ transmit({ action: "error", content: "Access denied" })
24
+ return
25
+ end
26
+ else
27
+ # Fallback to default if no encrypted path provided
28
+ file_path = Onlylogs.default_log_file_path
29
+ end
30
+ rescue Onlylogs::SecureFilePath::SecurityError => e
31
+ Rails.logger.error "Onlylogs: Security violation - #{e.message}"
32
+ transmit({ action: "error", content: "Access denied" })
33
+ return
34
+ end
35
+
36
+ # Check if the file is a text file
37
+ unless Onlylogs::File.text_file?(file_path)
38
+ transmit({ action: "error", content: "Cannot read file: File is not a text file" })
39
+ return
40
+ end
41
+
42
+ cursor_position = data["cursor_position"] || 0
43
+ filter = data["filter"].presence
44
+ mode = data["mode"] || "live"
45
+ regexp_mode = data["regexp_mode"] == true || data["regexp_mode"] == "true"
46
+ start_position = data["start_position"]&.to_i || 0
47
+ end_position = data["end_position"]&.to_i
48
+
49
+ if mode == "search"
50
+ # For search mode, read the entire file with filter and send all matching lines
51
+ read_entire_file_with_filter(file_path, filter, regexp_mode, start_position, end_position)
52
+ else
53
+ # For live mode, start the watcher
54
+ start_log_watcher(file_path, cursor_position, filter, regexp_mode)
55
+ end
56
+ end
57
+
58
+ def stop_watcher
59
+ cleanup_existing_operations
60
+ transmit({ action: "finish", content: "Search stopped." })
61
+ end
62
+
63
+ def unsubscribed
64
+ cleanup_existing_operations
65
+ end
66
+
67
+ private
68
+
69
+ def cleanup_existing_operations
70
+ if @batch_sender
71
+ @batch_sender.stop
72
+ @batch_sender = nil
73
+ end
74
+
75
+ stop_log_watcher
76
+ end
77
+
78
+ def start_log_watcher(file_path, cursor_position, filter = nil, regexp_mode = false)
79
+ return if @log_watcher_running
80
+
81
+ @log_watcher_running = true
82
+ @filter = filter
83
+ @regexp_mode = regexp_mode
84
+
85
+ transmit({ action: "message", content: "Reading file. Please wait..." })
86
+
87
+ @log_file = Onlylogs::File.new(file_path, last_position: cursor_position)
88
+
89
+ transmit({ action: "message", content: "" })
90
+
91
+ @log_watcher_thread = Thread.new do
92
+ Rails.logger.silence(Logger::ERROR) do
93
+ @log_file.watch do |new_lines|
94
+ break unless @log_watcher_running
95
+
96
+ # Collect all filtered lines from this batch
97
+ lines_to_send = []
98
+
99
+ new_lines.each do |log_line|
100
+ # Filters in live mode are not yet implemented
101
+ # if @filter.present? && !Onlylogs::Grep.match_line?(log_line.text, @filter, regexp_mode: @regexp_mode)
102
+ # next
103
+ # end
104
+
105
+ lines_to_send << {
106
+ line_number: log_line.number,
107
+ html: render_log_line(log_line)
108
+ }
109
+ end
110
+
111
+ if lines_to_send.any?
112
+ transmit({
113
+ action: "append_logs",
114
+ lines: lines_to_send
115
+ })
116
+ end
117
+ end
118
+ end
119
+ rescue StandardError => e
120
+ Rails.logger.error "Log watcher error: #{e.message}"
121
+ Rails.logger.error e.backtrace.join("\n")
122
+ ensure
123
+ @log_watcher_running = false
124
+ end
125
+ end
126
+
127
+ def stop_log_watcher
128
+ return unless @log_watcher_running
129
+
130
+ @log_watcher_running = false
131
+
132
+ return unless @log_watcher_thread&.alive?
133
+
134
+ @log_watcher_thread.kill
135
+ @log_watcher_thread.join(1)
136
+ end
137
+
138
+ def read_entire_file_with_filter(file_path, filter = nil, regexp_mode = false, start_position = 0, end_position = nil)
139
+ @log_watcher_running = true
140
+ @log_file = Onlylogs::File.new(file_path, last_position: 0)
141
+
142
+ transmit({ action: "message", content: "Searching..." })
143
+
144
+ @batch_sender = BatchSender.new(self)
145
+ @batch_sender.start
146
+
147
+ line_count = 0
148
+
149
+ Rails.logger.silence(Logger::ERROR) do
150
+ @log_file.grep(filter, regexp_mode: regexp_mode, start_position: start_position, end_position: end_position) do |log_line|
151
+ return if @batch_sender.nil?
152
+
153
+ # Add to batch buffer (sender thread will handle sending)
154
+ @batch_sender.add_line({
155
+ line_number: log_line.number,
156
+ html: render_log_line(log_line)
157
+ })
158
+
159
+ line_count += 1
160
+ end
161
+ end
162
+
163
+ # Stop batch sender and flush any remaining lines
164
+ @batch_sender.stop
165
+
166
+ # Send completion message
167
+ if line_count >= Onlylogs.max_line_matches
168
+ transmit({ action: "finish", content: "Search finished. Search results limit reached." })
169
+ else
170
+ transmit({ action: "finish", content: "Search finished." })
171
+ end
172
+
173
+ @log_watcher_running = false
174
+ end
175
+
176
+ def render_log_line(log_line)
177
+ "<pre data-line-number=\"#{log_line.number}\">" \
178
+ "<span class=\"line-number\">#{log_line.parsed_number}</span>#{log_line.parsed_text}</pre>"
179
+ end
180
+ end
181
+ end