PluginProbe ʕ •ᴥ•ʔ
AI Engine – The Chatbot, AI Framework & MCP for WordPress / 2.8.2
AI Engine – The Chatbot, AI Framework & MCP for WordPress v2.8.2
3.5.7 3.5.6 3.5.5 3.5.4 3.5.3 3.5.2 3.5.1 3.5.0 3.4.9 3.4.8 3.4.7 0.2.1 1.6.91 0.2.2 1.6.92 0.2.3 1.6.93 0.2.4 1.6.94 0.2.5 1.6.95 0.2.6 1.6.96 0.2.7 1.6.97 0.2.8 1.6.98 0.2.9 1.6.99 0.3.0 1.7.0 0.3.1 1.7.1 0.3.2 1.7.2 0.3.3 1.7.3 0.3.4 1.7.4 0.3.5 1.7.5 0.3.6 1.7.6 0.4.0 1.7.7 0.4.1 1.7.8 0.4.2 1.7.9 0.4.3 1.8.0 0.4.4 1.8.1 0.4.5 1.8.2 0.4.6 1.8.3 0.4.7 1.8.4 0.4.8 1.8.5 0.4.9 1.8.6 0.5.0 1.8.7 0.5.1 1.8.8 0.5.2 1.8.9 0.5.3 1.9.0 0.5.4 1.9.1 0.5.5 1.9.2 0.5.6 1.9.3 0.5.7 1.9.4 0.5.8 1.9.5 0.5.9 1.9.6 0.6.0 1.9.7 0.6.1 1.9.8 0.6.2 1.9.81 0.6.3 1.9.82 0.6.4 1.9.83 0.6.5 1.9.84 0.6.6 1.9.85 0.6.7 1.9.86 0.6.8 1.9.87 0.6.9 1.9.88 0.7.0 1.9.89 0.7.1 1.9.90 0.7.2 1.9.91 0.7.3 1.9.92 0.7.4 1.9.93 0.7.5 1.9.94 0.7.6 1.9.95 0.7.7 1.9.96 0.7.8 1.9.97 0.7.9 1.9.98 0.8.0 1.9.99 0.8.1 2.0.0 0.8.2 2.0.1 0.8.3 2.0.2 0.8.4 2.0.3 0.8.5 2.0.4 0.8.6 2.0.5 0.8.7 2.0.6 0.8.8 2.0.7 0.8.9 2.0.8 0.9.0 2.0.9 0.9.2 2.1.0 0.9.3 2.1.1 0.9.4 2.1.2 0.9.5 2.1.3 0.9.6 2.1.4 0.9.7 2.1.5 0.9.8 2.1.6 0.9.81 2.1.7 0.9.82 2.1.8 0.9.83 2.1.9 0.9.84 2.2.0 0.9.85 2.2.1 0.9.86 2.2.2 0.9.87 2.2.3 0.9.88 2.2.4 0.9.89 2.2.5 0.9.9 2.2.51 0.9.91 2.2.52 0.9.92 2.2.53 0.9.93 2.2.54 0.9.94 2.2.56 0.9.95 2.2.57 0.9.96 2.2.6 0.9.97 2.2.60 0.9.98 2.2.61 0.9.99 2.2.62 1.0.0 2.2.63 1.0.01 2.2.70 1.0.1 2.2.80 1.0.2 2.2.81 1.0.3 2.2.90 1.0.4 2.2.91 1.0.5 2.2.92 1.0.6 2.2.93 1.0.7 2.2.94 1.0.8 2.2.95 1.0.9 2.3.0 1.1.0 2.3.1 1.1.1 2.3.2 1.1.2 2.3.3 1.1.3 2.3.4 1.1.4 2.3.5 1.1.5 2.3.6 1.1.6 2.3.7 1.1.7 2.3.8 1.1.8 2.3.9 1.1.9 2.4.0 1.2.0 2.4.1 1.2.1 2.4.2 1.2.2 2.4.3 1.2.21 2.4.4 1.2.3 2.4.5 1.2.30 2.4.6 1.3.0 2.4.7 1.3.1 2.4.8 1.3.2 2.4.9 1.3.3 2.5.0 1.3.31 2.5.1 1.3.32 2.5.2 1.3.33 2.5.3 1.3.34 2.5.4 1.3.35 2.5.5 1.3.36 2.5.6 1.3.37 2.5.7 1.3.38 2.5.8 1.3.39 2.5.9 1.3.40 2.6.0 1.3.41 2.6.1 1.3.42 2.6.2 1.3.43 2.6.3 1.3.44 2.6.5 1.3.45 2.6.6 1.3.46 2.6.7 1.3.47 2.6.8 1.3.48 2.6.9 1.3.49 2.7.0 1.3.50 2.7.1 1.3.51 2.7.2 1.3.52 2.7.3 1.3.53 2.7.4 1.3.54 2.7.5 1.3.56 2.7.6 1.3.57 2.7.7 1.3.58 2.7.8 1.3.59 2.7.9 1.3.60 2.8.0 1.3.61 2.8.1 1.3.62 2.8.2 1.3.63 2.8.3 1.3.64 2.8.4 1.3.65 2.8.5 1.3.66 2.8.6 1.3.67 2.8.7 1.3.68 2.8.8 1.3.69 2.8.9 1.3.70 2.9.0 1.3.71 2.9.1 1.3.72 2.9.2 1.3.73 2.9.3 1.3.74 2.9.4 1.3.75 2.9.5 1.3.76 2.9.6 1.3.77 2.9.7 1.3.78 2.9.8 1.3.79 2.9.9 1.3.80 3.0.0 1.3.81 3.0.1 1.3.82 3.0.2 1.3.83 3.0.3 1.3.84 3.0.4 1.3.85 3.0.5 1.3.86 3.0.6 1.3.87 3.0.7 1.3.88 3.0.8 1.3.89 3.0.9 1.3.90 3.1.0 1.3.91 3.1.1 1.3.92 3.1.2 1.3.93 3.1.3 1.3.94 3.1.4 1.3.95 3.1.5 1.3.96 3.1.6 1.3.97 3.1.7 1.3.98 3.1.8 1.3.99 3.1.9 1.4.0 3.2.0 1.4.1 3.2.1 1.4.2 3.2.2 1.4.3 3.2.3 1.4.4 3.2.4 1.4.5 3.2.5 1.4.6 3.2.6 1.4.7 3.2.7 1.4.8 3.2.8 1.4.9 3.2.9 1.5.0 3.3.0 1.5.1 3.3.1 1.5.2 3.3.2 1.5.3 3.3.3 1.5.4 3.3.4 1.5.5 3.3.5 1.5.6 3.3.6 1.5.7 3.3.7 1.5.8 3.3.8 1.5.9 3.3.9 1.6.0 3.4.0 1.6.1 3.4.1 1.6.2 3.4.2 1.6.3 3.4.3 1.6.5 3.4.4 1.6.51 3.4.5 1.6.52 3.4.6 1.6.53 1.6.54 1.6.55 1.6.56 1.6.57 1.6.58 1.6.59 1.6.60 1.6.61 1.6.62 1.6.63 1.6.64 1.6.65 1.6.66 1.6.67 1.6.68 trunk 1.6.69 0.0.1 1.6.70 0.0.2 1.6.71 0.0.3 1.6.72 0.0.4 1.6.73 0.0.5 1.6.74 0.0.6 1.6.75 0.0.7 1.6.76 0.0.8 1.6.77 0.0.9 1.6.78 0.1.0 1.6.79 0.1.1 1.6.81 0.1.2 1.6.82 0.1.3 1.6.83 0.1.4 1.6.84 0.1.5 1.6.85 0.1.6 1.6.86 0.1.7 1.6.87 0.1.8 1.6.88 0.1.9 1.6.89 0.2.0 1.6.90
ai-engine / classes / logging.php
ai-engine / classes Last commit date
engines 1 year ago modules 1 year ago queries 1 year ago admin.php 1 year ago api.php 1 year ago core.php 1 year ago discussion.php 1 year ago init.php 1 year ago logging.php 1 year ago reply.php 1 year ago rest.php 1 year ago
logging.php
279 lines
1 <?php
2
3 /**
4 * Class Meow_MWAI_Logging
5 *
6 * A logging utility that uses the WordPress Filesystem API for storage,
7 * with fallback to PHP error_log when necessary.
8 */
9 class Meow_MWAI_Logging {
10 private static $plugin_name;
11 private static $option_name;
12 private static $log_file_path;
13 private static $fs;
14 private static $log_count = 0;
15 private static $rotate_check_frequency = 10;
16 private static $max_log_size = 5 * 1024 * 1024; // 5 MB
17
18 /**
19 * Initialize the logger.
20 *
21 * @param string $option_name Option key for settings.
22 * @param string $plugin_name Plugin identifier for error log prefix.
23 */
24 public static function init( $option_name, $plugin_name ) {
25 self::$plugin_name = $plugin_name;
26 self::$option_name = $option_name;
27
28 // Attempt to use WP_Filesystem only if the 'direct' method is available.
29 if ( ! function_exists( 'WP_Filesystem' ) ) {
30 require_once ABSPATH . 'wp-admin/includes/file.php';
31 }
32
33 if ( function_exists( 'get_filesystem_method' ) && 'direct' === get_filesystem_method() ) {
34 // If 'direct' is allowed, try to initialize the filesystem (no credentials prompt).
35 if ( WP_Filesystem() ) {
36 global $wp_filesystem;
37 self::$fs = $wp_filesystem;
38 } else {
39 // Could not initialize
40 error_log( self::$plugin_name . " : Could not init direct WP_Filesystem. Falling back to error_log only." );
41 self::$fs = null;
42 }
43 } else {
44 // Not 'direct' or not available; skip filesystem usage
45 self::$fs = null;
46 }
47
48 // Attempt to determine or create the log file path
49 self::$log_file_path = self::get_logs_path( true );
50 }
51
52 /**
53 * Determine or create the log file path using WP_Filesystem.
54 *
55 * @param bool $create Whether to generate a new file if none exists.
56 * @return string|false Path to log file or false if unavailable.
57 */
58 private static function get_logs_path( $create = false ) {
59 $options = get_option( self::$option_name, null );
60 if ( is_null( $options ) ) {
61 return null;
62 }
63
64 // If we don't have a filesystem reference, we can't create or write a file
65 if ( empty( self::$fs ) ) {
66 return null;
67 }
68
69 $path = empty( $options['logs_path'] ) ? '' : $options['logs_path'];
70
71 if ( $path && self::$fs->exists( $path ) ) {
72 return $path;
73 }
74
75 if ( ! $create ) {
76 return null;
77 }
78
79 $uploads = wp_upload_dir();
80 $base_dir = trailingslashit( $uploads['basedir'] );
81
82 if ( ! self::$fs->is_dir( $base_dir ) ) {
83 self::$fs->mkdir( $base_dir );
84 }
85
86 // Adjust MWAI_PREFIX to whatever your actual constant or value is
87 $filename = MWAI_PREFIX . '_' . self::random_ascii_chars() . '.log';
88 $new_path = $base_dir . $filename;
89
90 self::$fs->put_contents( $new_path, '', FS_CHMOD_FILE );
91
92 $options['logs_path'] = $new_path;
93 update_option( self::$option_name, $options );
94
95 return $new_path;
96 }
97
98 /**
99 * Check if logging is enabled via plugin options and FS availability.
100 *
101 * @return bool
102 */
103 private static function is_logging_enabled() {
104 $options = get_option( self::$option_name, null );
105 if ( is_null( $options ) ) {
106 return false;
107 }
108
109 $module_devtools = empty( $options['module_devtools'] ) ? false : $options['module_devtools'];
110 $server_debug_mode = empty( $options['server_debug_mode'] ) ? false : $options['server_debug_mode'];
111
112 return ( $module_devtools && $server_debug_mode && ! empty( self::$fs ) );
113 }
114
115 /**
116 * Internal log writer. Appends to file and/or error_log.
117 */
118 private static function add( $message = null, $icon = '', $error_log = false ) {
119 $date = date( 'Y-m-d H:i:s' );
120 $message = is_string( $message ) ? strip_tags( $message ) : $message;
121
122 if ( empty( $message ) ) {
123 $entry = "\n";
124 }
125 else if ( ! empty( $icon ) ) {
126 $entry = "$date: $icon $message\n";
127 }
128 else {
129 $entry = "$date: $message\n";
130 }
131
132 // Write to file if enabled and if a log file path exists
133 if ( self::is_logging_enabled() && self::$log_file_path ) {
134 if ( self::$fs->exists( self::$log_file_path ) ) {
135 $current = self::$fs->get_contents( self::$log_file_path );
136 self::$fs->put_contents( self::$log_file_path, $current . $entry, FS_CHMOD_FILE );
137 }
138 else {
139 self::$fs->put_contents( self::$log_file_path, $entry, FS_CHMOD_FILE );
140 }
141 }
142
143 // Always send to PHP error_log if $error_log is true
144 if ( $error_log && ! empty( $message ) ) {
145 \error_log( self::$plugin_name . " : $message" );
146 }
147
148 self::$log_count++;
149
150 if ( self::$log_count >= self::$rotate_check_frequency ) {
151 self::maybe_rotate_log();
152 self::$log_count = 0;
153 }
154 }
155
156 /**
157 * Logs a general message.
158 *
159 * @param string $message The message to log.
160 * @param string $icon Optional icon to prepend.
161 */
162 public static function log( $message = null, $icon = '' ) {
163 self::add( $message, $icon );
164 }
165
166 /**
167 * Logs a warning message.
168 *
169 * @param string $message The warning message to log.
170 * @param string $icon Optional icon to prepend (default ⚠️).
171 */
172 public static function warn( $message = null, $icon = '⚠️' ) {
173 self::add( $message, $icon );
174 }
175
176 /**
177 * Logs an error message and sends to PHP error_log.
178 *
179 * @param string $message The error message to log.
180 * @param string $icon Optional icon to prepend (default ❌).
181 */
182 public static function error( $message = null, $icon = '' ) {
183 self::add( $message, $icon, true );
184 }
185
186 /**
187 * Logs a deprecated feature notice.
188 *
189 * @param string $message The message to log.
190 */
191 public static function deprecated( $message = null ) {
192 self::add( $message, '🚨', true );
193 }
194
195 /**
196 * Clears the log file and resets the option.
197 */
198 public static function clear() {
199 if ( self::$fs && self::$log_file_path && self::$fs->exists( self::$log_file_path ) ) {
200 self::$fs->delete( self::$log_file_path );
201 $options = get_option( self::$option_name, null );
202 $options['logs_path'] = '';
203 update_option( self::$option_name, $options );
204 self::$log_file_path = '';
205 }
206 }
207
208 /**
209 * Retrieves the log contents in reverse order (newest first).
210 *
211 * @return string
212 */
213 public static function get() {
214 if ( self::$fs && self::$log_file_path && self::$fs->exists( self::$log_file_path ) ) {
215 $content = self::$fs->get_contents( self::$log_file_path );
216 $lines = explode( "\n", $content );
217 $lines = array_filter( $lines );
218 $lines = array_reverse( $lines );
219
220 return implode( "\n", $lines );
221 }
222
223 return 'Empty log file.';
224 }
225
226 /**
227 * Checks file size and rotates if exceeding maximum.
228 */
229 private static function maybe_rotate_log() {
230 if ( empty( self::$fs ) || empty( self::$log_file_path ) ) {
231 return;
232 }
233
234 if ( self::$fs->exists( self::$log_file_path ) ) {
235 $size = self::$fs->size( self::$log_file_path );
236
237 if ( $size > self::$max_log_size ) {
238 $info = pathinfo( self::$log_file_path );
239 $archived = $info['dirname'] . '/' . $info['filename'] . '_' . date( 'Y-m-d_H-i-s' ) . '.' . $info['extension'];
240
241 self::$fs->move( self::$log_file_path, $archived, true );
242 self::$fs->put_contents( self::$log_file_path, '', FS_CHMOD_FILE );
243 }
244 }
245 }
246
247 /**
248 * Generates a random ASCII string.
249 *
250 * @param int $length String length.
251 * @return string
252 */
253 private static function random_ascii_chars( $length = 8 ) {
254 $characters = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
255 $result = '';
256
257 for ( $i = 0; $i < $length; $i++ ) {
258 $result .= $characters[ mt_rand( 0, strlen( $characters ) - 1 ) ];
259 }
260
261 return $result;
262 }
263
264 /**
265 * Shortens a string to a specified length, adding ellipsis if needed.
266 *
267 * @param int $length String length.
268 * @return string
269 */
270 public static function shorten( $string, $length = 50 ) {
271 if ( strlen( $string ) > $length ) {
272 $string = rtrim( $string, " \t\n\r\0\x0B,." );
273 $string = substr( $string, 0, $length - 3 ) . '...';
274 }
275
276 return $string;
277 }
278 }
279